gpt4 book ai didi

android - 如何强制 RecyclerView.Adapter 在所有元素上调用 onBindViewHolder()

转载 作者:塔克拉玛干 更新时间:2023-11-02 22:35:54 25 4
gpt4 key购买 nike

我有一个 VerticalGridView,它使用 RecyclerView.Adapter 来填充元素。我发现如果潜在元素离开视口(viewport),则不会调用 onBindViewHolder() 方法。不幸的是,这导致了来自不同方法的 NullPointerException,因为我在 onBindViewHolder() 方法中捕获了一个 TextView 引用并将其传递给一个用于以后操作的外部变量。

@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
final ViewHolder viewHolder = (ViewHolder) holder;
viewHolder.txtCategoryName.setText(categories.get(position).getStrCategory());
categories.get(position).setTxtViewReference(viewHolder.txtCategoryDefectTotal);
viewHolder.categoryBoxRoot.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
for(CategoryListItem catItem : categories){
if(catItem.getStrCategory().equals(viewHolder.txtCategoryName.getText())){
int index = Defects.getInstance().getCategories().indexOf(catItem) + 1;
MainInterface.grids.get(index).bringToFront();
MainInterface.grids.get(index).setVisibility(View.VISIBLE);
for(VerticalGridView grid : MainInterface.grids){
int gridIndex = MainInterface.grids.indexOf(grid);
if(gridIndex != index){
grid.setVisibility(View.INVISIBLE);
}
}
break;
}
}

}
});

据我所知,在实例化 Viewholder 对象时会创建对 TextView 的引用。

public class ViewHolder extends RecyclerView.ViewHolder {

public TextView txtCategoryName;
public TextView txtCategoryDefectTotal;
public View categoryBoxRoot;

public ViewHolder(View itemView) {
super(itemView);
txtCategoryName = (TextView) itemView.findViewById(R.id.textViewCategoryName);
txtCategoryDefectTotal = (TextView) itemView.findViewById(R.id.textViewCategoryTotalDefects);
categoryBoxRoot = itemView.findViewById(R.id.root_category_box);
}
}

有没有一种方法可以在 Adapter 实例化时强制在所有元素上至少调用一次 onBindViewHolder()

我尝试了这些建议 here没有任何成功。

我知道在所有元素上强制执行 onBindViewHolder() 会违反 whole purpose of the RecycleView.Adapter .因此,我愿意接受有关如何捕获该 TextView 引用的任何其他建议。

作为此问题的临时解决方案,我可以在生成 NullPointerException 的方法周围使用 try catch block 。但是,我担心缺少引用意味着我将来可能会引入错误。

最佳答案

这个问题最简单的解决方案是向下滚动到网格底部,然后再回到顶部。这不能在 onCreate() 方法中完成,因为当时网格在技术上是不可见的。相反,它应该在 Activity 的 onResume() 方法中调用。

@Override
public void onResume(){
super.onResume();
VerticalGridView defectGrid = grids.get(0);
RecyclerView.Adapter adapter = defectGrid.getAdapter();
defectGrid.smoothScrollToPosition(adapter.getItemCount()-1);
defectGrid.smoothScrollToPosition(0);
}

这是一个很好的解决方案尝试,但不幸的是它仍然不起作用。虽然它确实获得了使该方法起作用的引用,但它不一定具有正确的引用。我发现引用最终可能指向不同的 TextView,因为 RecycleView.Adapter 重用 View 以在不同区域显示它们。

解决方案:

Mark Keen 说使用 notifyDataSetChanged() 是对的。我通过修复我的 onBindViewHolder() 方法使其正常工作。

@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
final ViewHolder viewHolder = (ViewHolder) holder;
viewHolder.txtCategoryName.setText(categories.get(position).getStrCategory());
viewHolder.txtCategoryDefectTotal.setText(String.valueOf(categories.get(position).getTotalDefectsInCategory()));
}

我还更改了我的数据对象,因此它包含一个 int 值而不是对 TextView 的引用,因为上面证明引用是无效的。最后,当我按下自定义后退按钮时,我添加了对我的 Adapter 的调用。

grids.get(0).getAdapter().notifyDataSetChanged();

感谢大家的贡献!

关于android - 如何强制 RecyclerView.Adapter 在所有元素上调用 onBindViewHolder(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38130686/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com