在 RecyclerView 适配器 notifyDataSetChanged() 之后未调用 onBindViewHolder

Posted

技术标签:

【中文标题】在 RecyclerView 适配器 notifyDataSetChanged() 之后未调用 onBindViewHolder【英文标题】:onBindViewHolder not called after RecyclerView Adapter notifyDataSetChanged() 【发布时间】:2019-01-15 10:20:34 【问题描述】:

我正在从 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_
          android:layout_
          android:orientation="vertical">

<LinearLayout
    android:layout_
    android:layout_
    android:orientation="horizontal"
    android:weightSum="2">

    <TextView
        android:layout_
        android:layout_
        android:layout_weight="1"
        android:gravity="center"
        android:text="USERNAME"
        android:textColor="#000"
        android:textSize="16sp"/>

    <TextView
        android:layout_
        android:layout_
        android:layout_weight="1"
        android:gravity="center"
        android:text="SCORE"
        android:textColor="#000"
        android:textSize="16sp"/>

</LinearLayout>

<View
    android:layout_
    android:layout_
    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_
                                        android:layout_
                                        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_
          android:layout_
          android:orientation="vertical">

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_
              android:layout_
              android:orientation="horizontal"
              android:weightSum="2">

    <TextView
        android:id="@+id/username"
        android:layout_
        android:layout_
        android:layout_weight="1"
        android:gravity="center"
        android:textSize="14sp"/>

    <TextView
        android:id="@+id/score"
        android:layout_
        android:layout_
        android:layout_weight="1"
        android:gravity="center"
        android:textSize="14sp"/>


</LinearLayout>

<View
    android:layout_
    android:layout_
    android:layout_marginEnd="10dp"
    android:layout_marginStart="10dp"
    android:background="#4545"/>

所以它将包含两个水平的 TextViews 和一个 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 中设置文本(用户名和分数)

最后是我试图通知适配器的活动:

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) 

                                                       
                                                   

    );


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

但是在调用 notifyDataSetChanged() 之后,屏幕只是闪烁,一切都是空白的……在 onBindViewHolder 中设置断点后无法访问方法……知道这里有什么问题吗?

【问题讨论】:

在适配器中,总是在每个 if 语句中使用 else。特别是在使用视图时 请不要包含您的屏幕图片。几乎不可能阅读。如果您确实必须包含图像,则应使用屏幕截图。 另外,您的 POJO 类似乎缺少 default constructor 对不起,我要找一个朋友,这就是她发给我的。另外,它也不适用于默认构造函数 【参考方案1】:

您的代码中有几个问题。第一个是您正在调用setResults() 方法并将results 列表传递给ScoreAdapter 类和not 到构造函数。为了解决这个问题,将你的 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 列表传递给构造函数。

另外请不要忘记将public no-argument constructor 添加到您的Result2 课程中。请参阅here了解更多详情。

【讨论】:

Same sh** again -.- ...我不认为应该有任何区别,因为适配器应该等待数据,然后它调用 notifyDataSetChanged().. 另外,没有参数也提供了构造函数......太奇怪了 适配器没有等待。试试我的解决方案,看看它是否有效。 我做到了,完全一样...:/ 那是什么行为?还是一样的黑屏? 这很奇怪,但你见过这个post吗?

以上是关于在 RecyclerView 适配器 notifyDataSetChanged() 之后未调用 onBindViewHolder的主要内容,如果未能解决你的问题,请参考以下文章

在 RecyclerView 的适配器中实现的 OnClickListener 不起作用

片段中的recyclerview的“RecyclerView:没有附加适配器;跳过布局”[重复]

如何更新 RecyclerView 适配器数据

在 RecyclerView 适配器 notifyDataSetChanged() 之后未调用 onBindViewHolder

RecyclerView 适配器中的 onBackPressed()

RecyclerView