数据从 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 中
这可能不是您期望的输出顺序。正如您所看到的行after回调之前被调用onDataChange
。这解释了为什么您返回的列表是空的,或者(更正确地)当您返回它时它是空的并且稍后才会被填充。
有几种方法可以处理这种异步加载。
最简单的解释就是把所有返回列表的代码放在一起into the 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>)
}
现在您可以将此接口的实现传递给您的getContactsFromFirebase
method:
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")
}
})
它不像同步加载数据那么简单,但是这样做的优点是它运行时不会阻塞主线程。
这个话题已经讨论过了a lot之前,所以我建议您也检查其中一些问题:
- this 道格的博客文章
-
在 Firebase Listener 中设置 Singleton 属性值(我解释了在某些情况下你如何can获得同步数据加载,但通常不能)
-
返回一个对象 Android(我第一次用日志语句来解释发生了什么)
- 是否可以从 Firebase 同步加载数据?
-
https://stackoverflow.com/a/38188683(Doug 展示了一种很酷但复杂的使用方法
Task
API 与 Firebase 数据库)
-
如何返回 DataSnapshot 值作为方法的结果?(我从那里借用了一些回调语法)