ArrayList不在onChildAdded函数内更新

时间:2016-09-13 07:57:49

标签: java android firebase firebase-realtime-database

public class DataService {

    private static DataService ourInstance = new DataService();
    private DatabaseReference mDatabase;


    public static DataService getInstance() {
        return ourInstance;
    }

    public ArrayList<UserDatabase> getFriendList() {

        mDatabase = FirebaseDatabase.getInstance().getReference().child("users");
        final ArrayList<UserDatabase> list = new ArrayList<>();

        mDatabase.addChildEventListener(new ChildEventListener() {
            @Override
            public void onChildAdded(DataSnapshot dataSnapshot, String s) {
                UserDatabase userDatabase = dataSnapshot.getValue(UserDatabase.class);
                list.add(userDatabase);
            }

            @Override
            public void onChildChanged(DataSnapshot dataSnapshot, String s) {}

            @Override
            public void onChildRemoved(DataSnapshot dataSnapshot) {}

            @Override
            public void onChildMoved(DataSnapshot dataSnapshot, String s) {}

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

        return list;
     }
}

我正在尝试从users节点获取用户数据库。它的表现非常奇怪。当我在onChildAdded函数设置断点时,它将检索所有用户并将其添加到列表中。

但是,如果我没有在那里设置断点,那么它没有得到任何东西,列表大小为0.有没有人知道发生了什么?提前谢谢!

1 个答案:

答案 0 :(得分:6)

数据是从Firebase异步加载和同步的。

如果您添加一些适当的日志记录语句,最简单的方法是查看导致问题的原因:

public ArrayList<UserDatabase> getFriendList() {

    mDatabase = FirebaseDatabase.getInstance().getReference().child("users");
    final ArrayList<UserDatabase> list = new ArrayList<>();

    System.out.println("Attaching listener");
    mDatabase.addChildEventListener(new ChildEventListener() {
        @Override
        public void onChildAdded(DataSnapshot dataSnapshot, String s) {
            System.out.println("Got data");
        }

        public void onChildChanged(DataSnapshot dataSnapshot, String s) {}
        public void onChildRemoved(DataSnapshot dataSnapshot) {}
        public void onChildMoved(DataSnapshot dataSnapshot, String s) {}
        public void onCancelled(DatabaseError databaseError) {}
    });

    System.out.println("Returning list");
    return list;
 }

输出将按此顺序打印:

  

附加监听器

     

返回列表

     

获得数据

这可能不是你预期的顺序。但它确实解释了为什么如果你按原样运行代码,那么列表为空。

原因是Firebase异步加载和同步数据。像大多数现代Web API一样,它不会等待结果返回(这会导致可怕的用户体验),而是让程序继续并在获取(新)数据时回调代码。

处理这种行为的方法是重新编写程序流程,从“首先获取朋友列表,然后使用它进行xyz”到“每当朋友列表发生变化时,用它做xyz”。

例如说你只想打印有多少朋友。您当前的方法可能是:

ArrayList<UserDatabase> friends = getFriendList();
System.out.println("There are "+friends.size()+" friends");

使用Firebase处理此问题的一种方法是将操作移至 getFriendList()

public void getFriendList() {

    mDatabase = FirebaseDatabase.getInstance().getReference().child("users");
    final ArrayList<UserDatabase> list = new ArrayList<>();

    mDatabase.addChildEventListener(new ChildEventListener() {
        @Override
        public void onChildAdded(DataSnapshot dataSnapshot, String s) {
            UserDatabase userDatabase = dataSnapshot.getValue(UserDatabase.class);
            list.add(userDatabase);
            System.out.println("There are "+friends.size()+" friends");
        }

        public void onChildChanged(DataSnapshot dataSnapshot, String s) {}
        public void onChildRemoved(DataSnapshot dataSnapshot) {}
        public void onChildMoved(DataSnapshot dataSnapshot, String s) {}
        public void onCancelled(DatabaseError databaseError) {}
    });

 }