如何更新RecyclerView适配器数据

我试图弄清楚更新RecyclerView的适配器的问题是什么。

在我得到一个新的产品清单后,我试着:

  1. 从创建recyclerView的片段中更新ArrayList,将新数据设置为适配器,然后调用adapter.notifyDataSetChanged();这并没有起作用。

  2. 创建一个新的适配器,就像其他人做的那样,它为他们工作,但对我来说没有任何变化:recyclerView.setAdapter(new RecyclerViewAdapter(newArrayList))

  3. Adapter中创建一个更新数据的方法,如下所示:

     public void updateData(ArrayList<ViewModel> viewModels) {
    items.clear();
    items.addAll(viewModels);
    notifyDataSetChanged();
    }
    

    然后每当我想要更新数据列表时,我就调用这个方法;这并没有起作用。

  4. 检查我是否可以修改recyclerView以任何方式,我尝试删除至少一个项目:

      public void removeItem(int position) {
    items.remove(position);
    notifyItemRemoved(position);
    }
    

一切都保持原样。

这是我的适配器:

public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder> implements View.OnClickListener {


private ArrayList<ViewModel> items;
private OnItemClickListener onItemClickListener;


public RecyclerViewAdapter(ArrayList<ViewModel> items) {
this.items = items;
}




public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
this.onItemClickListener = onItemClickListener;
}


@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_recycler, parent, false);
v.setOnClickListener(this);
return new ViewHolder(v);
}


public void updateData(ArrayList<ViewModel> viewModels) {
items.clear();
items.addAll(viewModels);
notifyDataSetChanged();
}
public void addItem(int position, ViewModel viewModel) {
items.add(position, viewModel);
notifyItemInserted(position);
}


public void removeItem(int position) {
items.remove(position);
notifyItemRemoved(position);
}




@Override
public void onBindViewHolder(ViewHolder holder, int position) {
ViewModel item = items.get(position);
holder.title.setText(item.getTitle());
Picasso.with(holder.image.getContext()).load(item.getImage()).into(holder.image);
holder.price.setText(item.getPrice());
holder.credit.setText(item.getCredit());
holder.description.setText(item.getDescription());


holder.itemView.setTag(item);
}




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




@Override
public void onClick(final View v) {
// Give some time to the ripple to finish the effect
if (onItemClickListener != null) {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
onItemClickListener.onItemClick(v, (ViewModel) v.getTag());
}
}, 0);
}
}


protected static class ViewHolder extends RecyclerView.ViewHolder {
public ImageView image;
public TextView price, credit, title, description;


public ViewHolder(View itemView) {
super(itemView);
image = (ImageView) itemView.findViewById(R.id.image);
price = (TextView) itemView.findViewById(R.id.price);
credit = (TextView) itemView.findViewById(R.id.credit);
title = (TextView) itemView.findViewById(R.id.title);
description = (TextView) itemView.findViewById(R.id.description);
}
}


public interface OnItemClickListener {


void onItemClick(View view, ViewModel viewModel);


}
}

并且我像下面这样初始化RecyclerView:

recyclerView = (RecyclerView) view.findViewById(R.id.recycler);
recyclerView.setLayoutManager(new GridLayoutManager(getActivity(), 5));
adapter = new RecyclerViewAdapter(items);
adapter.setOnItemClickListener(this);
recyclerView.setAdapter(adapter);

那么,如何实际更新适配器数据以显示新接收到的项呢?


问题是gridView的布局是这样的:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:tag="catalog_fragment"
android:layout_height="match_parent">


<FrameLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">


<android.support.v7.widget.RecyclerView
android:id="@+id/recycler"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clipToPadding="false"/>


<ImageButton
android:id="@+id/fab"
android:layout_gravity="top|end"
style="@style/FabStyle"/>


</FrameLayout>
</LinearLayout>

然后我删除了LinearLayout,并将FrameLayout作为父布局。

669053 次浏览

我正在使用RecyclerView和删除和更新工作得很好。

  1. < p > <强>删除< / >强:

    从RecyclerView中删除项目有四个步骤

     list.remove(position);
    recycler.removeViewAt(position);
    mAdapter.notifyItemRemoved(position);
    mAdapter.notifyItemRangeChanged(position, list.size());
    

    这几行代码对我有用。

  2. < p > 更新数据:

    我唯一要做的就是:

     mAdapter.notifyDataSetChanged();
    

您必须在活动/片段代码中完成所有这些,而不是在RecyclerView适配器代码中。

这对我来说很管用:

recyclerView.setAdapter(new RecyclerViewAdapter(newList));
recyclerView.invalidate();

在创建一个包含更新列表的新适配器(在我的情况下,它是一个转换为数组列表的数据库)并将其设置为适配器之后,我尝试了recyclerView.invalidate(),它工作了。

向现有数据添加新数据的最佳和最酷的方法是

 ArrayList<String> newItems = new ArrayList<String>();
newItems = getList();
int oldListItemscount = alcontainerDetails.size();
alcontainerDetails.addAll(newItems);
recyclerview.getAdapter().notifyItemChanged(oldListItemscount+1, al_containerDetails);

你有两个选择:

从适配器刷新UI:

mAdapter.notifyDataSetChanged();

或者从recyclerView本身刷新它:

recyclerView.invalidate();
我用不同的方法解决了同样的问题。 我没有数据。我在后台线程等待它,所以从一个空列表开始
mAdapter = new ModelAdapter(getContext(), new ArrayList<Model>());
// Then when I get data


mAdapter.update(response.getModelList());
// And update it in my adapter


public void update(ArrayList<Model> modelList){
adapterModelList.clear();
for (Product model: modelList) {
adapterModelList.add(model);
}
mAdapter.notifyDataSetChanged();
}

就是这样。

另一个选项是使用diffutil。它将把原始列表与新列表进行比较,如果有更改,则使用新列表作为更新。

基本上,我们可以使用DiffUtil来比较旧数据和新数据,并让它代表你调用notifyItemRangeRemoved, notifyItemRangeChanged和notifyItemRangeInserted。

一个使用diffUtil代替notifyDataSetChanged的快速示例:

DiffResult diffResult = DiffUtil
.calculateDiff(new MyDiffUtilCB(getItems(), items));


//any clear up on memory here and then
diffResult.dispatchUpdatesTo(this);


//and then, if necessary
items.clear()
items.addAll(newItems)

我在主线程外做calculateDiff工作,以防它是一个使用AsyncListDiffer的大列表

更新listview, gridview和recyclerview的数据:

mAdapter.notifyDataSetChanged();

或者:

mAdapter.notifyItemRangeChanged(0, itemList.size());

过了很久,我得到了答案:

SELECTEDROW.add(dt);
notifyItemInserted(position);
SELECTEDROW.remove(position);
notifyItemRemoved(position);

如果上面的评论对你不起作用,这可能意味着问题出在其他地方。

我发现解决方案的一个地方是我将列表设置到适配器的方式。在我的活动中,列表是一个实例变量,当任何数据发生变化时,我直接改变它。由于它是一个参考变量,所以发生了一些奇怪的事情。

因此,我将引用变量更改为本地变量,并使用另一个变量更新数据,然后传递给前面回答中提到的addAll()函数。

这些方法是有效的,开始使用一个基本的RecyclerView

private List<YourItem> items;


public void setItems(List<YourItem> newItems)
{
clearItems();
addItems(newItems);
}


public void addItem(YourItem item, int position)
{
if (position > items.size()) return;


items.add(item);
notifyItemInserted(position);
}


public void addMoreItems(List<YourItem> newItems)
{
int position = items.size() + 1;
newItems.addAll(newItems);
notifyItemChanged(position, newItems);
}


public void addItems(List<YourItem> newItems)
{
items.addAll(newItems);
notifyDataSetChanged();
}


public void clearItems()
{
items.clear();
notifyDataSetChanged();
}


public void addLoader()
{
items.add(null);
notifyItemInserted(items.size() - 1);
}


public void removeLoader()
{
items.remove(items.size() - 1);
notifyItemRemoved(items.size());
}


public void removeItem(int position)
{
if (position >= items.size()) return;


items.remove(position);
notifyItemRemoved(position);
}


public void swapItems(int positionA, int positionB)
{
if (positionA > items.size()) return;
if (positionB > items.size()) return;


YourItem firstItem = items.get(positionA);


videoList.set(positionA, items.get(positionB));
videoList.set(positionB, firstItem);


notifyDataSetChanged();
}

你可以在一个适配器类或你的片段或活动中实现它们,但在这种情况下,你必须实例化适配器来调用通知方法。在我的情况下,我通常在适配器中实现它。

这是一个普遍的答案。本文解释了更新适配器数据的各种方法。该过程每次包括两个主要步骤:

  1. 更新数据集
  2. 将更改通知适配器

插入单个项目

添加“Pig"在索引2

Insert single item
String item = "Pig";
int insertIndex = 2;
data.add(insertIndex, item);
adapter.notifyItemInserted(insertIndex);

插入多个项目

在索引2处插入另外三只动物。

Insert multiple items
ArrayList<String> items = new ArrayList<>();
items.add("Pig");
items.add("Chicken");
items.add("Dog");
int insertIndex = 2;
data.addAll(insertIndex, items);
adapter.notifyItemRangeInserted(insertIndex, items.size());

删除一个项目

删除“Pig"从名单上。

Remove single item
int removeIndex = 2;
data.remove(removeIndex);
adapter.notifyItemRemoved(removeIndex);

移除多个项目

删除“Camel"和“;Sheep"从名单上。

Remove multiple items
int startIndex = 2; // inclusive
int endIndex = 4;   // exclusive
int count = endIndex - startIndex; // 2 items will be removed
data.subList(startIndex, endIndex).clear();
adapter.notifyItemRangeRemoved(startIndex, count);

移除所有物品

清除整个列表。

Remove all items
data.clear();
adapter.notifyDataSetChanged();

用新列表替换旧列表

清除旧列表,然后添加一个新列表。

Replace old list with new list
// clear old list
data.clear();


// add new list
ArrayList<String> newList = new ArrayList<>();
newList.add("Lion");
newList.add("Wolf");
newList.add("Bear");
data.addAll(newList);


// notify adapter
adapter.notifyDataSetChanged();

adapter有一个对data的引用,所以我没有将data设置为一个新对象是很重要的。相反,我清除了data中的旧项,然后添加了新的项。

更新单项

改变“Sheep"这样它就会说“我喜欢羊”;

Update single item
String newValue = "I like sheep.";
int updateIndex = 3;
data.set(updateIndex, newValue);
adapter.notifyItemChanged(updateIndex);

移动单个物品

“移动;Sheep"从位置3到位置1

Move single item
int fromPosition = 3;
int toPosition = 1;


// update data array
String item = data.get(fromPosition);
data.remove(fromPosition);
data.add(toPosition, item);


// notify adapter
adapter.notifyItemMoved(fromPosition, toPosition);

代码

这里是项目代码供您参考。RecyclerView适配器代码可以在这个答案找到。

MainActivity.java

public class MainActivity extends AppCompatActivity implements MyRecyclerViewAdapter.ItemClickListener {


List<String> data;
MyRecyclerViewAdapter adapter;


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


// data to populate the RecyclerView with
data = new ArrayList<>();
data.add("Horse");
data.add("Cow");
data.add("Camel");
data.add("Sheep");
data.add("Goat");


// set up the RecyclerView
RecyclerView recyclerView = findViewById(R.id.rvAnimals);
LinearLayoutManager layoutManager = new LinearLayoutManager(this);
recyclerView.setLayoutManager(layoutManager);
DividerItemDecoration dividerItemDecoration = new DividerItemDecoration(recyclerView.getContext(),
layoutManager.getOrientation());
recyclerView.addItemDecoration(dividerItemDecoration);
adapter = new MyRecyclerViewAdapter(this, data);
adapter.setClickListener(this);
recyclerView.setAdapter(adapter);
}


@Override
public void onItemClick(View view, int position) {
Toast.makeText(this, "You clicked " + adapter.getItem(position) + " on row number " + position, Toast.LENGTH_SHORT).show();
}


public void onButtonClick(View view) {
insertSingleItem();
}


private void insertSingleItem() {
String item = "Pig";
int insertIndex = 2;
data.add(insertIndex, item);
adapter.notifyItemInserted(insertIndex);
}


private void insertMultipleItems() {
ArrayList<String> items = new ArrayList<>();
items.add("Pig");
items.add("Chicken");
items.add("Dog");
int insertIndex = 2;
data.addAll(insertIndex, items);
adapter.notifyItemRangeInserted(insertIndex, items.size());
}


private void removeSingleItem() {
int removeIndex = 2;
data.remove(removeIndex);
adapter.notifyItemRemoved(removeIndex);
}


private void removeMultipleItems() {
int startIndex = 2; // inclusive
int endIndex = 4;   // exclusive
int count = endIndex - startIndex; // 2 items will be removed
data.subList(startIndex, endIndex).clear();
adapter.notifyItemRangeRemoved(startIndex, count);
}


private void removeAllItems() {
data.clear();
adapter.notifyDataSetChanged();
}


private void replaceOldListWithNewList() {
// clear old list
data.clear();


// add new list
ArrayList<String> newList = new ArrayList<>();
newList.add("Lion");
newList.add("Wolf");
newList.add("Bear");
data.addAll(newList);


// notify adapter
adapter.notifyDataSetChanged();
}


private void updateSingleItem() {
String newValue = "I like sheep.";
int updateIndex = 3;
data.set(updateIndex, newValue);
adapter.notifyItemChanged(updateIndex);
}


private void moveSingleItem() {
int fromPosition = 3;
int toPosition = 1;


// update data array
String item = data.get(fromPosition);
data.remove(fromPosition);
data.add(toPosition, item);


// notify adapter
adapter.notifyItemMoved(fromPosition, toPosition);
}
}

笔记

  • 如果使用notifyDataSetChanged(),则不会执行动画。这也可能是一个昂贵的操作,所以如果你只更新一个项或一组项,不建议使用notifyDataSetChanged()
  • 检查DiffUtil如果你正在对一个列表做大或复杂的改变。

进一步的研究

  • CodePath: Using the RecyclerView
  • < a href = " https://android.jlelse。eu/ Smart -way-to-update- RecyclerView -using- DiffUtil -345941a160e0" rel="noreferrer">使用DiffUtil更新RecyclerView的智能方式

我发现重载RecyclerView的一个很简单的方法就是调用

recyclerView.removeAllViews();

这将首先删除RecyclerView的所有内容,然后用更新后的值再次添加它。

我强烈建议你使用[DiffUtil.ItemCallback][1]来处理RecyclerView.Adapter中的变化:

fun setData(data: List<T>) {
val calculateDiff = DiffUtil.calculateDiff(DiffUtilCallback(items, data))
items.clear()
items += data
calculateDiff.dispatchUpdatesTo(this)
}

在底层,它用AdapterListUpdateCallback处理大部分事情:

/**
* ListUpdateCallback that dispatches update events to the given adapter.
*
* @see DiffUtil.DiffResult#dispatchUpdatesTo(RecyclerView.Adapter)
*/
public final class AdapterListUpdateCallback implements ListUpdateCallback {
@NonNull
private final RecyclerView.Adapter mAdapter;


/**
* Creates an AdapterListUpdateCallback that will dispatch update events to the given adapter.
*
* @param adapter The Adapter to send updates to.
*/
public AdapterListUpdateCallback(@NonNull RecyclerView.Adapter adapter) {
mAdapter = adapter;
}


/** {@inheritDoc} */
@Override
public void onInserted(int position, int count) {
mAdapter.notifyItemRangeInserted(position, count);
}


/** {@inheritDoc} */
@Override
public void onRemoved(int position, int count) {
mAdapter.notifyItemRangeRemoved(position, count);
}


/** {@inheritDoc} */
@Override
public void onMoved(int fromPosition, int toPosition) {
mAdapter.notifyItemMoved(fromPosition, toPosition);
}


/** {@inheritDoc} */
@Override
public void onChanged(int position, int count, Object payload) {
mAdapter.notifyItemRangeChanged(position, count, payload);
}
}


对于Kotlin,我会使用The Adapter,

class ProductAdapter(var apples: List<Apples>?= null) : RecyclerView.Adapter<ProductViewHolder>() {
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ProductViewHolder {...}


override fun onBindViewHolder(holder: ProductViewHolder, position: Int) {...}


override fun getItemCount(): Int {...}


fun setData(newApples: List<Apples>) {
apples = newApples
}
}

在片段/活动

val appleAdapter = ProductAdapter()


val recyclerView = binding.appleRecycler // or findViewById or synthetics or whatever.
recyclerView.adapter = appleAdapter
recyclerView.layoutManager = LinearLayoutManager(requireContext())

现在处理片段中的数据更改

fun updateRecyclerData(apples: List<Apples>){
adapter.setData(apples)
adapter.notifyDataSetChanged()
}


updateRecyclerData(applesList)

这个很好试试。

  ArrayList.remove(position);
notifyItemRemoved(position);
notifyDataSetChanged();

我建议你去探索DiffUtil。它还提高了RecyclerView在处理列表更新时的性能。

  1. 在适配器中定义一个变量:

     differList = AsyncListDiffer(this, this.callback);
    differList.submitList(list)
    

在这里,列表可以是您最初的原始列表,也可以只是一个空列表,前提是您稍后将更新它。

  1. < p >实现回调:

    private val callback : DiffUtil.ItemCallback<Item> = object: DiffUtil.ItemCallback<Item>() {
    
    
    override fun areItemsTheSame(oldItem: Item, newItem: Item) =
    oldItem.id == newItem.id
    
    
    override fun areContentsTheSame(oldItem: Item, newItem: Item) =
    oldItem == newItem
    }
    
  2. 同样,在同一个适配器中,您将有一些公共函数来设置列表。

     fun setData(list: List<Item>) {
    differList.submitList(list)
    //yeah, that's it!
    }
    
  1. 现在,在你对列表做了任何更改(插入/更新/删除)之后,只需从你的fragment/activity中调用这个setData(列表)

    mAdapter.setData(list)

容易,对吧?