Android:等待firebase valueEventListener

Dav*_*uan 5 android asynchronous semaphore firebase firebase-realtime-database

我正在尝试使用Semaphore等待我的firebase valueEventListener.我有一个用户信息活动,有6个不同的字段,用户必须填写.当用户保存他/她的信息时,我想进行"全有或全无"类型的检查.某些用户信息无法复制...例如用户名,电子邮件和电话号码.我正在使用firebase,目前一般的想法是这样的格式:

void saveUserInfo(){
    if(field1 exist in database){
        return;
    }
    .
    .
    .
    if(field6 exist in database){
        return;
    }

    savefield1();
    .
    .
    .
    savefield6();
}
Run Code Online (Sandbox Code Playgroud)

我遇到的问题是检查数据库中是否已存在该值的方法.这是我目前的方法:

   public boolean alreadyInUse(String key, String value) throws InterruptedException {

    final StringBuilder done = new StringBuilder("");
    final Semaphore semaphore = new Semaphore(0);

    mDatabase.child(key).child(value).addListenerForSingleValueEvent(new ValueEventListener() {

        @Override
        public void onDataChange(DataSnapshot dataSnapshot) {
            String result = dataSnapshot.getValue(String.class);
            if(result == null){
                Log.d("WorkoutLog", "result: null");
                done.append("false");
                semaphore.release();
                return;
            }
            else if(result.equals(uID)){
                Log.d("WorkoutLog", "result: " + result.toString());
                done.append("false");
                semaphore.release();
                return;
            }
            else{
                Log.d("WorkoutLog", "result: " + result.toString());
                done.append("true");
                semaphore.release();
                return;
            }

        }

        @Override
        public void onCancelled(DatabaseError databaseError) {

        }
    });


    semaphore.acquire();
    if(done.equals("true")){
        return  true;
    }
    else if(done.equals("false")){
        return false;
    }
    else{
        Log.d("WorkoutLog", "Shouldn't be here");
        return true;
    }

}
Run Code Online (Sandbox Code Playgroud)

现在信号量还没有释放......想知道是否有人可以帮助我.没有信号量,return语句将在firebase查询完成之前触发...

JP *_*ura 12

如前所述,您可以使用Google I/O 2016上提供的Firebase Task API.

Task<?>[] tasks = new Task[] {
    saveUserName(username),
    saveFriends(friends),
    saveOtherStuff(stuff)
};

Tasks.whenAll(tasks)
    .continueWithTask(new RollbackIfFailure())
    .addOnCompleteListener(this)
    .addOnFailureListener(this);
Run Code Online (Sandbox Code Playgroud)

如果进行修改的每个步骤可以并行运行,则可以创建一个方法,Task为每个步骤返回一个,如下所示:

public Task<String> saveUserName(final String username) {
    final TaskCompletionSource<String> tcs = new TaskCompletionSource<>();

    mDatabase.child("users")
        .child(username)
        .addListenerForSingleValueEvent(new ValueEventListener() {
        @Override
        public void onDataChange(DataSnapshot dataSnapshot) {
            String username = dataSnapshot.getValue(String.class);
            if (null == username) {
                tcs.setResult(null);
            } else {
                tcs.setException(new IllegalStateException(username));
            }
        }

        @Override
        public void onCancelled(FirebaseError firebaseError) {
            tcs.setException(new IOException(TAG, firebaseError.toException()));
        }
    });

    return tcs.getTask();
}
Run Code Online (Sandbox Code Playgroud)

如果其中任何一个失败,则需要回滚操作.由于这可以通过单个任务进行,因此您可以创建Continuation任务:

class RollbackIfFailure implements Continuation<Void, Task<Void>> {
    @Override
    public Task<Void> then(@NonNull Task<Void> task) throws Exception {

        final TaskCompletionSource<Void> tcs = new TaskCompletionSource<>();

        if (task.isSuccessful()) {
            tcs.setResult(null);
        } else {
            // Rollback everything
        }

        return tcs.getTask();
    }
}
Run Code Online (Sandbox Code Playgroud)


Bob*_*der 7

侦听器回调方法在主线程上运行.如果你调用alreadyInUse()主线程,它将阻塞线程semaphore.acquire(),防止回调运行和释放信号量.

您可能会发现对相关问题的答案很有帮助.