- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
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
这可能不是您期望的输出顺序。如您所见,after 行,回调在 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")
}
})
它不像同步加载数据那样简单,但它的优点是它可以在不阻塞主线程的情况下运行。
这个主题之前已经讨论过很多,所以我建议您也查看其中的一些问题:
Task
API 与 Firebase 数据库结合使用的很酷但很复杂的方法)关于android - getContactsFromFirebase() 方法返回一个空列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53896418/
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
public List getContactsFromFirebase(){ FirebaseDatabase.getInstance().getReference().child("User
我是一名优秀的程序员,十分优秀!