如何使用 Firebase 查询的结果最终填充 Android Studio 中的 ListView

How to use results from a Firebase query to ultimately populate a ListView in Android Studio

我发现 我的 ListView 在应用 运行 时没有更新,因为适配器是在 Firebase 查询发生之前设置的。

第一个问题:有没有办法为查询完成设置监听器?这样我就可以在那里设置 adapter 了。我知道一些 Firebase 方法会自动生成 onComplete()/onSuccess() 听众。

出于调试目的,我可以通过重新输入 onResume() 来触发对 ListView 的更新。这是我收到错误消息的地方:

The content of the adapter has changed but ListView did not receive a notification. Make sure the content of your adapter is not modified from a background thread, but only from the UI thread. Make sure your adapter calls notifyDataSetChanged() when its content changes. [in ListView(16908298, class android.widget.ListView) with Adapter(class android.widget.ArrayAdapter)]

据我了解 这个错误意味着我无法在 onChildAdded() 方法中向我的 ArrayListusernames” 添加项目,如果我使用usernames 在我的 ArrayAdapter 中。

第二个问题:我还能如何存储从我的 Firebase 查询返回的值并使用这些值最终填充 ListView 而无需直接从 "background thread" 更改其内容?

我的尝试:

public class EditFriendsActivity extends AppCompatActivity {

    protected ArrayList<String> usernames = new ArrayList<>();

    @Override
    protected void onResume() {
        super.onResume();

        Firebase usernameRef = ref.child("users");
        Query queryRef = usernameRef.orderByKey(); 

        queryRef.addChildEventListener(new ChildEventListener() {
        @Override
        public void onChildAdded(DataSnapshot dataSnapshot, String s) {
            String username = (String) dataSnapshot.child("username").getValue(); 
            usernames.add(username); 
            Log.v(TAG, usernames + "");
        }
        @Override
        public void onChildChanged(DataSnapshot dataSnapshot, String s) {
            Log.v(TAG, "Inside ChildChanged");
        }

        @Override
        public void onChildRemoved(DataSnapshot dataSnapshot) {
            Log.v(TAG, "Inside ChildRemoved");
        }

        @Override
        public void onChildMoved(DataSnapshot dataSnapshot, String s) {
            Log.v(TAG, "Inside ChildMoved");
        }

        @Override
        public void onCancelled(FirebaseError firebaseError) {

            Log.e(TAG, firebaseError.getMessage());

            AlertDialog.Builder builder = new AlertDialog.Builder(EditFriendsActivity.this);
            builder.setTitle(R.string.error_title)
            .setMessage(firebaseError.getMessage())
            .setPositiveButton(android.R.string.ok, null);

            AlertDialog dialog = builder.create();
            dialog.show();
            }
        });

        // Array adapter
        ArrayAdapter<String> adapter = new ArrayAdapter<>(
                EditFriendsActivity.this,
                android.R.layout.simple_list_item_multiple_choice, 
                usernames);

        mFriendsList.setAdapter(adapter); 
    }
}  

更新代码: 修复了每次标记答案和清除列表中的侦听器中的拼写错误。

Firebase userRef = ref.child("users");
userRef.addListenerForSingleValueEvent(new ValueEventListener() {

    @Override
    public void onDataChange(DataSnapshot dataSnapshot) {
        usernames.clear(); // Clear list before updating
        for (DataSnapshot child : dataSnapshot.getChildren()) {
            String username = (String) child.child("username").getValue();
            usernames.add(username);
            Log.v(TAG, usernames + "");
        }        

跟进问题Here

Is there anyway to set a listener for query completion? This way I can set the adapter there instead. I know some Firebase methods auto-generate onComplete()/onSuccess() listeners.

Firebases 将数据从后端同步到您的应用程序。因此,与具有 request/response 流的传统数据库不同,Firebase 查询并未真正完成。这就是为什么在 FirebaseUI library 中我们保持开放的侦听器并通知 ListView 任何更改。

也就是说:如果你只想从一个位置加载一次数据,你可以使用addSingleValueEventListener:

queryRef.addListenerForSingleValueEvent(new ValueEventListener() {
    @Override
    public void onDataChange(DataSnapshot snapshot) {
        for (DataSnapshot child: snapshot.getChildren()) {
            String username = (String) child.child("username").getValue(); 
            usernames.add(username); 
        }
        ArrayAdapter<String> adapter = new ArrayAdapter<>(
            EditFriendsActivity.this,
            android.R.layout.simple_list_item_multiple_choice, 
            usernames);

        mFriendsList.setAdapter(adapter); 
    }

    @Override
    public void onCancelled(FirebaseError firebaseError) {
        Log.e(TAG, firebaseError.getMessage());
    }
});

上面可能有错别字,但大致意思是这样。

您可以简单地使用查询

Query ref = FirebaseDatabase.getInstance().getReference()
            .child("users").orderByChild("name");