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

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


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


  附加监听器之前
  
  附加监听器后
  
  在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);
}


科特林:

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


现在,您可以将此接口的实现传递给您的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();
    }
  });
}


科特林:

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


科特林:

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


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

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


这个blog post from Doug
Setting Singleton property value in Firebase Listener(在这里我解释了在某些情况下如何获得同步数据加载,但通常无法做到)
return an object Android(我第一次使用日志语句解释发生了什么事)
Is it possible to synchronously load data from Firebase?
https://stackoverflow.com/a/38188683(其中Doug显示了将Task API与Firebase数据库一起使用的一种酷而复杂的方法)
How to return DataSnapshot value as a result of a method?(从这里借用了一些回调语法)

07-27 19:19