getContactsFromFirebase()方法返回一个空列表

Pan*_*007 3 android firebase firebase-realtime-database

public List<String> getContactsFromFirebase(){
    FirebaseDatabase.getInstance().getReference().child("Users")
            .addListenerForSingleValueEvent(new ValueEventListener() {
                @Override
                public void onDataChange(DataSnapshot dataSnapshot) {
                    for (DataSnapshot snapshot : dataSnapshot.getChildren()) {
                        Users user = snapshot.getValue(Users.class);
                        assert user != null;
                        String contact_found = user.getPhone_number();
                        mContactsFromFirebase.add(contact_found);
                        Log.i("Test", mContactsFromFirebase.toString());
                    }

                }
                @Override
                public void onCancelled(DatabaseError databaseError) {
                }
            });

    return mContactsFromFirebase;

}
Run Code Online (Sandbox Code Playgroud)

我似乎无法找到错误.在上面的代码中,当我调用日志时,我从中获取值mContactsFromFirebase,但该getContactsFromFirebase()方法返回一个空列表.请问你能帮帮我吗?

Fra*_*len 9

数据是从Firebase异步加载的.由于从服务器获取数据可能需要一些时间,因此主要Android代码会继续,Firebase会onDataChange在数据可用时调用您的数据.

这意味着当return mContactsFromFirebase它仍然是空的时候.最简单的方法是查看几个日志语句:

System.out.println("Before attaching listener");
FirebaseDatabase.getInstance().getReference().child("Users")
    .addListenerForSingleValueEvent(new ValueEventListener() {
      @Override
      public void onDataChange(DataSnapshot dataSnapshot) {
        System.out.println("In onDataChange");
      }
      @Override
      public void onCancelled(DatabaseError databaseError) {
        throw databaseError.toException(); // don't ignore errors
      }
    });
System.out.println("After attaching listener");
Run Code Online (Sandbox Code Playgroud)

当您运行此代码时,它将打印:

在附加监听器之前

附加监听器后

在onDataChange中

这可能不是您期望输出的顺序.正如您可以看到之前调用回调之后的行onDataChange.这就解释了为什么你返回的列表是空的,或者(更正确地)当你返回它时它是空的并且只是稍后填充.

有几种方法可以处理这种异步加载.

解释最简单的就是把它返回列表中的所有代码onDataChange方法.这意味着此代码仅在加载数据后执行.最简单的形式:

public void onDataChange(DataSnapshot dataSnapshot) {
    for (DataSnapshot snapshot : dataSnapshot.getChildren()) {
        Users user = snapshot.getValue(Users.class);
        assert user != null;
        String contact_found = user.getPhone_number();
        mContactsFromFirebase.add(contact_found);
        System.out.println("Loaded "+mContactsFromFirebase.size()+" contacts");
    }
}
Run Code Online (Sandbox Code Playgroud)

但是有更多的方法,包括使用自定义回调(类似于Firebase自己的回调ValueEventListener):

public interface UserListCallback {
  void onCallback(List<Users> value);
}
Run Code Online (Sandbox Code Playgroud)

现在,您可以将此接口的实现传递给您的getContactsFromFirebase方法:

public void getContactsFromFirebase(final UserListCallback myCallback) {
  databaseReference.child(String.format("users/%s/name", uid)).addListenerForSingleValueEvent(new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
      for (DataSnapshot snapshot : dataSnapshot.getChildren()) {
        Users user = snapshot.getValue(Users.class);
        assert user != null;
        String contact_found = user.getPhone_number();
        mContactsFromFirebase.add(contact_found);
        System.out.println("Loaded "+mContactsFromFirebase.size()+" contacts");
      }
      myCallback.onCallback(mContactsFromFirebase);
    }

    @Override
    public void onCancelled(DatabaseError databaseError) {
      throw databaseError.toException();
    }
  });
}
Run Code Online (Sandbox Code Playgroud)

然后像这样调用它:

getContactsFromFirebase(new UserListCallback() {
  @Override
  public void onCallback(List<Users> users) {
    System.out.println("Loaded "+users.size()+" contacts")
  }
});
Run Code Online (Sandbox Code Playgroud)

它并不像同步加载数据那么简单,但这样做的好处是它可以在不阻塞主线程的情况下运行.

之前已经讨论过很多这个话题,所以我建议你也看看其中的一些问题:


归档时间:

查看次数:

891 次

最近记录:

6 年,6 月 前