Как обновить данные адаптера RecyclerView?
В течение 4 дней я не могу понять, что именно происходит с обновлением адаптера RecyclerView
.
После получения нового списка продуктов я попытался:
-
Обновите ArrayList
из фрагмента, где RecyclerView
создан, установите новые данные в адаптер, затем вызовите adapter.notifyDataSetChanged()
; это не сработало.
-
Создайте новый адаптер, как и другие, и он сработал для них, но никаких изменений для меня: recyclerView.setAdapter(new RecyclerViewAdapter(newArrayList))
-
Создайте метод в Adapter
, который обновляет данные следующим образом:
public void updateData(ArrayList<ViewModel> viewModels) {
items.clear();
items.addAll(viewModels);
notifyDataSetChanged();
}
Затем я вызываю этот метод всякий раз, когда я хочу обновить список данных; это не работает.
-
Чтобы проверить, могу ли я каким-либо образом изменить 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
в качестве родительского макета.
Ответы
Ответ 1
Я работаю с RecyclerView, и работа по удалению и обновлению хорошо работает.
1) УДАЛИТЬ:
Есть 4 шага для удаления элемента из RecyclerView
list.remove(position);
recycler.removeViewAt(position);
mAdapter.notifyItemRemoved(position);
mAdapter.notifyItemRangeChanged(position, list.size());
Эта строка кодов работает для меня.
2) ОБНОВЛЕНИЕ ДАННЫХ:
Единственное, что мне нужно было сделать, это
mAdapter.notifyDataSetChanged();
Вам нужно было сделать все это в коде Actvity/Fragment, но не в коде адаптера RecyclerView.
Ответ 2
Это то, что сработало для меня:
recyclerView.setAdapter(new RecyclerViewAdapter(newList));
recyclerView.invalidate();
После создания нового адаптера, который содержит обновленный список (в моем случае это была база данных, преобразованная в ArrayList) и установив ее как адаптер, я попробовал recyclerView.invalidate()
и работал.
Ответ 3
у вас есть 2 варианта:
обновить пользовательский интерфейс от адаптера:
mAdapter.notifyDataSetChanged();
или обновить его из самого recyclerView:
recyclerView.invalidate();
Ответ 4
Другой вариант - использовать 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);
Я делаю calclateDiff, работающий с основным потоком, если это большой список.
Ответ 5
Обновить данные списка, gridview и recyclerview
mAdapter.notifyDataSetChanged();
или
mAdapter.notifyItemRangeChanged(0, itemList.size());
Ответ 6
Я решил ту же проблему по-другому.
У меня нет данных, которые я ожидаю от фонового потока, поэтому начинайте с списка emty.
mAdapter = new ModelAdapter(getContext(),new ArrayList<Model>());
// then when i get data
mAdapter.update(response.getModelList());
// and update is in my adapter
public void update(ArrayList<Model> modelList){
adapterModelList.clear();
for (Product model: modelList) {
adapterModelList.add(model);
}
mAdapter.notifyDataSetChanged();
}
Что это.
Ответ 7
Самый лучший и самый крутой способ добавления новых данных к данным -
ArrayList<String> newItems = new ArrayList<String>();
newItems = getList();
int oldListItemscount = alcontainerDetails.size();
alcontainerDetails.addAll(newItems);
recyclerview.getAdapter().notifyItemChanged(oldListItemscount+1, al_containerDetails);
Ответ 8
Я получил ответ через долгое время
SELECTEDROW.add(dt);
notifyItemInserted(position);
SELECTEDROW.remove(position);
notifyItemRemoved(position);
Ответ 9
Если в приведенных выше комментариях ничего не говорится, вы будете работать. Это может означать, что проблема лежит где-то в другом месте.
В одном месте я нашел решение в том, как я настраивал список для адаптера. В моей деятельности список был переменной экземпляра, и я менял его непосредственно, когда какие-либо данные изменялись. Из-за того, что это контрольная переменная, было что-то странное. Поэтому я изменил ссылочную переменную на локальную и использовал другую переменную для обновления данных, а затем перешел к функции addAll()
, упомянутой в приведенных выше ответах.
Ответ 10
Это общий ответ для будущих посетителей. Объясняются различные способы обновления данных адаптера. Процесс включает в себя два основных этапа каждый раз:
- Обновление набора данных
- Уведомлять об адаптере изменения
Вставить отдельный элемент
Добавьте "Свинья" в индекс 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());
Удалить один элемент
Удалите из списка "Свиньи".
![Remove single item]()
int removeIndex = 2;
data.remove(removeIndex);
adapter.notifyItemRemoved(removeIndex);
Удалить несколько элементов
Удалите из списка "Верблюд" и "Овцы".
![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
, а затем добавил новые.
Обновить один элемент
Измените пункт "Овцы", чтобы он произносил "Я люблю овец".
![Update single item]()
String newValue = "I like sheep.";
int updateIndex = 3;
data.set(updateIndex, newValue);
adapter.notifyItemChanged(updateIndex);
Переместить отдельный элемент
Переместите "Овец" из положения 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, если вы делаете большие или сложные изменения в списке.
Дальнейшее изучение
Ответ 11
Эти методы эффективны и полезны для начала использования базового 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();
}
Вы можете реализовать их внутри класса адаптера или вашего фрагмента или действия, но в этом случае вам необходимо создать экземпляр адаптера для вызова методов уведомления. В моем случае я обычно реализую его в адаптере.
Ответ 12
Я вижу все эти методы, которые требуют времени, я нашел способ.
Я просто включил Intent и назвал завершение, это гарантирует, что список
обновилась.
Intent intent = new Intent(context,thisActivity.class);
startActivity(intent);
finish();
Готово!