【问题标题】:Android: wait for firebase valueEventListenerAndroid:等待firebase valueEventListener
【发布时间】:2016-12-22 07:21:33
【问题描述】:

我正在尝试使用 Semaphore 来等待我的 firebase valueEventListener。我有一个用户信息活动,其中包含用户必须填写的 6 个不同字段。当用户保存他/她的信息时,我想做一个“全有或全无”类型的检查。某些用户信息不能重复...例如用户名、电子邮件和电话号码。我正在使用 firebase,目前的总体思路是:

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

    savefield1();
    .
    .
    .
    savefield6();
}

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

   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;
    }

}

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

【问题讨论】:

标签: android asynchronous firebase firebase-realtime-database semaphore


【解决方案1】:

如前所述,您可以使用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);

如果进行修改的每个步骤都可以并行运行,您可以创建一个方法,为每个步骤返回一个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();
}

如果其中任何一个失败,您需要回滚操作。由于这可能会受到单个任务的威胁,因此您可以创建一个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();
    }
}

【讨论】:

    【解决方案2】:

    侦听器回调方法在主线程上运行。如果您在主线程上调用alreadyInUse(),它将阻塞semaphore.acquire() 处的线程,从而阻止回调运行并释放信号量。

    您可能会发现this answer 对相关问题很有帮助。

    【讨论】:

      猜你喜欢
      • 2018-08-16
      • 1970-01-01
      • 2018-09-23
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-04-16
      相关资源
      最近更新 更多