在RecyclerView Adapter调用notifyDataSetChanged()后,onBindViewHolder方法未被调用

8

我正在从Firebase数据库获取一些数据,并尝试使用它来填充RecyclerView适配器。在调用Adapter的notifyDataSetChanged()之后,屏幕会闪烁,但什么也不会发生,我甚至无法在onBindViewHolder中捕获断点。

以下是我的代码:

POJO类:

public class Result2 implements Serializable {

private int score;
private String userName;

public Result2(int score, String userName) {
    this.score = score;
    this.userName = userName;
}

public int getScore() {
    return score;
}


public String getUserName() {
    return userName;
}

这是我的活动布局,名为activity_results.xml,其中包含RecyclerView。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
          android:layout_width="match_parent"
          android:layout_height="match_parent"
          android:orientation="vertical">

<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="50dp"
    android:orientation="horizontal"
    android:weightSum="2">

    <TextView
        android:layout_width="0dp"
        android:layout_height="match_parent"
        android:layout_weight="1"
        android:gravity="center"
        android:text="USERNAME"
        android:textColor="#000"
        android:textSize="16sp"/>

    <TextView
        android:layout_width="0dp"
        android:layout_height="match_parent"
        android:layout_weight="1"
        android:gravity="center"
        android:text="SCORE"
        android:textColor="#000"
        android:textSize="16sp"/>

</LinearLayout>

<View
    android:layout_width="match_parent"
    android:layout_height="1dp"
    android:background="#000"
    />

<android.support.v7.widget.RecyclerView xmlns:android="http://schemas.android.com/apk/res/android"
                                        xmlns:app="http://schemas.android.com/apk/res-auto"
                                        android:id="@+id/recycler_view"
                                        android:layout_width="match_parent"
                                        android:layout_height="match_parent"
                                        android:clipToPadding="false"
                                        android:paddingTop="10dp"
                                        android:scrollbars="vertical"
                                        app:layout_behavior="@string/appbar_scrolling_view_behavior"/>

这是我的适配器ViewHolder布局,名为score_view_holder.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
          android:layout_width="match_parent"
          android:layout_height="wrap_content"
          android:orientation="vertical">

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="50dp"
              android:orientation="horizontal"
              android:weightSum="2">

    <TextView
        android:id="@+id/username"
        android:layout_width="0dp"
        android:layout_height="match_parent"
        android:layout_weight="1"
        android:gravity="center"
        android:textSize="14sp"/>

    <TextView
        android:id="@+id/score"
        android:layout_width="0dp"
        android:layout_height="match_parent"
        android:layout_weight="1"
        android:gravity="center"
        android:textSize="14sp"/>


</LinearLayout>

<View
    android:layout_width="match_parent"
    android:layout_height="1dp"
    android:layout_marginEnd="10dp"
    android:layout_marginStart="10dp"
    android:background="#4545"/>

需要展示两个水平的 TextView 和一条 View 作为分隔线。

这是我的适配器:

public class ScoreAdapter extends RecyclerView.Adapter<ScoreAdapter.ScoreViewHolder> {

private List<Result2> results = new ArrayList<>();

public void setResults(List<Result2> results) {
    this.results.clear();
    this.results.addAll(results);
    notifyDataSetChanged();
}

@Override
public ScoreAdapter.ScoreViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.score_view_holder, parent, false);
    return new ScoreAdapter.ScoreViewHolder(view);
}

@Override
public void onBindViewHolder(ScoreAdapter.ScoreViewHolder holder, int position) {
    Result2 result = getResult(position);
    if (result != null) {
        holder.setUsername(result.getUserName() != null ? result.getUserName() : "-");
        holder.setScore(String.valueOf(result.getScore()));
    }
}

private Result2 getResult(int position) {
    return !results.isEmpty() ? results.get(position) : null;
}

@Override
public int getItemCount() {
    return results.size();
}

public class ScoreViewHolder extends RecyclerView.ViewHolder {

    private TextView username;
    private TextView score;

    public ScoreViewHolder(View itemView) {
        super(itemView);
        username = itemView.findViewById(R.id.username);
        score = itemView.findViewById(R.id.score);
    }

    public void setUsername(String username) {
        this.username.setText(username);
    }

    public void setScore(String score) {
        this.score.setText(score);
    }

}
}

它应该获取Result2对象的列表,并仅在这两个TextView中设置文本(用户名和分数)

最后,我的Activity尝试从中通知适配器:

public class Results extends AppCompatActivity {

private DatabaseReference mDatabase;
private ScoreAdapter scoreAdapter;
private RecyclerView recyclerView;
private List<Result2> results = new ArrayList<>();


@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_results);

    recyclerView = findViewById(R.id.recycler_view);
    scoreAdapter = new ScoreAdapter();
    recyclerView.setLayoutManager(new LinearLayoutManager(this));
    recyclerView.setAdapter(scoreAdapter);

    mDatabase = FirebaseDatabase.getInstance().getReference();

    loadResults();
}

private void loadResults() {

    mDatabase.child("Users").addValueEventListener(new ValueEventListener() {
                                                       @Override
                                                       public void onDataChange(@NonNull DataSnapshot dataSnapshot) {

                                                           for (DataSnapshot childSnapshot : dataSnapshot.getChildren()) {
                                                               Result2 result = childSnapshot.getValue(Result2.class);

                                                               if (result != null) {
                                                                   results.add(result);
                                                               }
                                                           }
                                                           Toast.makeText(Results.this, String.valueOf(results.size()), Toast.LENGTH_SHORT).show();
                                                           scoreAdapter.setResults(results);

                                                       }

                                                       @Override
                                                       public void onCancelled(@NonNull DatabaseError databaseError) {

                                                       }
                                                   }

    );
}
}

循环完成后,Toast显示正确的结果数量,调用适配器的setResults方法,我收到了正确的结果数量,在这里有一张图片:

List of results inside setResults()

但是,在调用notifyDataSetChanged()后,屏幕会闪烁并且一切都变空白… 在onBindViewHolder中的方法无法被调用,即使在它们上面设置断点… 你知道这里有什么问题吗?

在适配器中,始终在每个if语句中使用else。特别是在处理视图时。 - Amroun
请不要包含屏幕截图,因为这几乎不可能阅读。如果您确实需要包含图像,则应使用屏幕截图。 - André Kool
似乎你的POJO类缺少一个默认构造函数 - André Kool
抱歉,我是代朋友提问的,这就是她发给我的内容。同时,使用默认构造函数也无法解决问题。 - joe
2个回答

0

你的代码存在几个问题。第一个问题是你在ScoreAdapter类中调用了setResults()方法并将你的results列表传递进去,而不是传递给构造函数。为了解决这个问题,你需要将setResults()方法改成构造函数,像这样:

public ScoreAdapter(List<Result2> results) {
    this.results.clear();
    this.results.addAll(results);
    notifyDataSetChanged();
}

适配器也必须在回调函数中实例化并设置到你的RecyclerView中。因此,为了解决这个问题,只需将以下代码行移动即可:
ScoreAdapter scoreAdapter = new ScoreAdapter(result);
recyclerView.setAdapter(scoreAdapter);

在下面这行代码之后:

Toast.makeText(Results.this, String.valueOf(results.size()), Toast.LENGTH_SHORT).show();

别忘了注释这行代码:scoreAdapter.setResults(results);

现在看看,要实例化适配器,你需要将result列表传递给构造函数。

还请不要忘记为您的Result2类添加public no-argument constructor。请参见此处获取更多详细信息。


1
又是同样的破事 -.- ... 我不相信会有任何区别,因为适配器应该等待数据,然后调用notifyDataSetChanged().. 另外,还提供了无参构造函数... 太奇怪了。 - joe
适配器没有等待。尝试我的解决方案,看看是否有效。 - Alex Mamo
我确实是这样做的... :/ - joe
行为是什么?还是同样的空白屏幕? - Alex Mamo
这很奇怪,但是你看过这个**帖子**吗? - Alex Mamo
1
一切都好吗,你解决了问题吗? - Alex Mamo

0
我遇到了同样的问题,当我调用notifyDataSetChanged时,onBindViewHolder没有被调用。 只有当我触摸并尝试滚动时,onBindViewHolder才会被调用。
我通过在通知适配器后添加一个小滚动来解决我的问题:
binding.rvwContent.scrollBy(0,1) // this is scroll down

嘿,欢迎来到StackOverflow。关于回答格式的几点建议:你可以使用反引号 \`` 来突出显示代码片段、方法名等,使其以等宽字体显示,像这样`。此外,在提及方法名时,请尽量准确地使用大小写。当你写下 "onbindviewholder" 时,我能理解你的意思,但对于那些对API还不太熟悉的人来说,可能会导致错误,并且阅读起来更加困难。感谢你的贡献,希望你能继续保持。 - Vaz

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接