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

标签 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;

}

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

最佳答案

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

这意味着当您返回 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");

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

Before attaching listener

After attaching listener

In 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");
    }
}
<小时/>

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

Java:

public interface UserListCallback {
  void onCallback(List<Users> value);
}

Kotlin :

interface UserListCallback {
  fun onCallback(value:List<Users>)
}

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

Java:

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

Kotlin :

fun getContactsFromFirebase(myCallback:UserListCallback) {
  databaseReference.child(String.format("users/%s/name", uid)).addListenerForSingleValueEvent(object:ValueEventListener() {
    fun onDataChange(dataSnapshot:DataSnapshot) {
      for (snapshot in dataSnapshot.getChildren())
      {
        val user = snapshot.getValue(Users::class.java)
        assert(user != null)
        val contact_found = user.getPhone_number()
        mContactsFromFirebase.add(contact_found)
        System.out.println("Loaded " + mContactsFromFirebase.size() + " contacts")
      }
      myCallback.onCallback(mContactsFromFirebase)
    }
    fun onCancelled(databaseError:DatabaseError) {
      throw databaseError.toException()
    }
  })

然后这样调用它:

Java:

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

Kotlin :

getContactsFromFirebase(object:UserListCallback() {
  fun onCallback(users:List<Users>) {
    System.out.println("Loaded " + users.size() + " contacts")
  }
})

它不像同步加载数据那么简单,但是这样做的优点是它运行时不会阻塞主线程。

这个主题之前已经讨论过很多次了,所以我建议您也查看其中的一些问题:

关于android - getContactsFromFirebase() 方法返回一个空列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58899429/

相关文章:

android - 在 Android 中从 Firebase 实时数据库中检索数据

java - 这些对象会被GC回收吗?

node.js - Firebase 云消息传递 : Request contains an invalid argument

android - Google Play 服务每次都会对版本发出警告

node.js - Firebase 部署在 tsc 命令中返回错误

android - 如何通过一笔交易在 Firebase 上创建用户并上传数据

firebase - 我应该将 firebase api key 隐藏到后端吗?不是因为数据安全,而是项目克隆问题

java - 在 Realtime Firebase 中从数据库检索数据时出现问题

android - java.lang.NoClassDefFoundError :failed resolution of :Lorg/apache/http/ProtocolVersion

android - 如何让 Eclipse/ADT 支持我添加到 build.xml 的 nocompress 标签?