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

标签 java android android-listview android-studio firebase

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

第一个问题:是否有办法为查询完成设置监听器?这样我就可以在那里设置适配器。我知道一些 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” code> 方法,如果我在 ArrayAdapter 中使用 usernames

第二个问题:我还能如何存储从 Firebase 查询返回的值并使用这些值最终填充 ListView,而不直接从“后台线程”更改其内容?

我的尝试:

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.

Firebase 将数据从后端同步到您的应用。因此,与具有请求/响应流的传统数据库不同,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());
    }
});

上面可能有错别字,但这是大致要点。

关于java - 如何使用 Firebase 查询的结果最终填充 Android Studio 中的 ListView,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32511133/

相关文章:

java - 项目 euler 23(循环)出现问题

javascript - 在 Play Framework 上将参数从 View 传递到 Controller

java - 在 JSON 字符串中添加双引号的简单方法

android - 没有AlertDialog的android中的多选微调器

java - 在底部导航 View 上多次点击时应用程序崩溃

java - 任务 ':app:mergeDebugResources'的执行失败。 > java.util.concurrent.ExecutionException:com.android.builder.internal.aapt.v2.Aapt2Exception:

android - WebRTC android-PeerConnectionFactory.createPeerConnection 始终返回 null

安卓 : How to dismiss a DialogFragment on orientation change?

Android "smoother"滚动时工具栏隐藏

android - 如何避免将 Fragment 重新创建到 ViewPager 中?