代码之家  ›  专栏  ›  技术社区  ›  joe

在RecyclerView适配器NotifyDataSetChanged()之后未调用OnBindViewHolder

  •  0
  • joe  · 技术社区  · 6 年前

    我正在从firebase数据库中获取一些数据,并尝试填充 RecyclerView 适配器与它。适配器之后 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;
    }
    

    }

    这是我的activitys布局,名为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"/>
    

    这是我的适配器view holder布局,名为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"/>
    

    因此它将包含两个水平文本视图和一个作为下面一行的视图。

    这是我的适配器:

    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对象的列表,并在这两个文本视图中设置文本(用户名和分数)

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

    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显示正确的结果数,adapter setresults方法被调用,这里我收到正确的结果数,这里是一张图片:

    List of results inside setResults()

    但在调用notifydatasetChanged()之后,屏幕会闪烁,并且所有内容都是空白的…在OnBindViewHolder中放置断点后,无法访问其中的方法。你知道这里怎么了吗?

    1 回复  |  直到 6 年前
        1
  •  0
  •   Alex Mamo    6 年前

    代码中存在几个问题。第一个是你打电话给 setResults() 方法并通过 results 在你的 ScoreAdapter 类和 给施工方。为了解决这个问题,把你的 StRESULTS() 方法成为这样的构造函数:

    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 更多细节。

    推荐文章