Попытка выяснить, в чем проблема с обновлением 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
как родительский макет.
источник
items.clear(); items.addAll(newItems);
это уродливый шаблон. Если вам действительно нужно защитное копирование здесь,items = new ArrayList(newItems);
это будет менее уродливо.Ответы:
Я работаю с RecyclerView, и удаление, и обновление работают хорошо.
1) УДАЛИТЬ: есть 4 шага, чтобы удалить элемент из RecyclerView
Эти строки кодов работают для меня.
2) ОБНОВЛЕНИЕ ДАННЫХ: единственное, что мне нужно было сделать, это
Вы должны были сделать все это в коде Actvity / Fragment, а не в коде RecyclerView Adapter.
источник
recycler.removeViewAt(position);
(точнееrecycler.removeAllViews()
) были решающими.Это общий ответ для будущих посетителей. Описаны различные способы обновления данных адаптера. Процесс включает в себя два основных шага каждый раз:
Вставить один элемент
Добавить "Свинья" в указателе
2
.Вставьте несколько предметов
Вставьте еще трех животных в указатель
2
.Удалить один элемент
Удалить «Свинью» из списка.
Удалить несколько предметов
Уберите «Верблюд» и «Овцу» из списка.
Удалить все предметы
Очистить весь список.
Заменить старый список новым списком
Очистите старый список, затем добавьте новый.
adapter
Имеет ссылкуdata
, так что важно , что я не ставилdata
на новый объект. Вместо этого я очистил старые элементы,data
а затем добавил новые.Обновить один элемент
Измените пункт «Овцы», чтобы он говорил «Я люблю овец».
Переместить один элемент
Переместите «Овцу» из положения
3
в положение1
.Код
Вот код проекта для вашей справки. Код адаптера RecyclerView можно найти в этом ответе .
MainActivity.java
Ноты
notifyDataSetChanged()
, то анимация не будет выполняться. Это также может быть дорогостоящей операцией, поэтому ее не рекомендуется использовать,notifyDataSetChanged()
если вы обновляете только один элемент или диапазон элементов.Дальнейшее обучение
источник
Update single item
, должно быть, я люблю черепахВот что сработало для меня:
После создания нового адаптера, который содержит обновленный список (в моем случае это была база данных, преобразованная в ArrayList) и установки его в качестве адаптера, я попытался,
recyclerView.invalidate()
и это сработало.источник
YourAdapter adapter = (YourAdapter) recyclerView.getAdapter(); adapter.yourSetterMethod(newList); adapter.notifyDataSetChanged();
Это, как говорится», это звучит как то, что сначала попробовал ОП (просто добавив это в качестве комментария, чтобы это могло помочь кому-то еще, как это сработало в моем случае).у вас есть 2 варианта: обновить интерфейс от адаптера:
или обновите его из себя в recyclerView:
источник
Другим вариантом является использование diffutil . Он будет сравнивать исходный список с новым списком и использовать новый список в качестве обновления, если есть изменения.
По сути, мы можем использовать DiffUtil для сравнения старых данных с новыми данными и позволить ему вызывать notifyItemRangeRemoved, а также notifyItemRangeChanged и notifyItemRangeInserted от вашего имени.
Быстрый пример использования diffUtil вместо notifyDataSetChanged:
Я работаю с CalculateDiff от основного потока на случай, если это большой список.
источник
getItems()
. Как вы узнаете, какие данные из исходного списка необходимо удалить / добавить / обновить?Обновление данных списка, вида сетки и повторного просмотра
или
источник
Лучший и самый крутой способ добавления новых данных к текущим данным
источник
Я решил ту же проблему по-другому. У меня нет данных, которые я жду от фонового потока, поэтому начнем со списка пустых файлов.
Вот и все.
источник
Эти методы эффективны и хороши, чтобы начать использовать основные
RecyclerView
.Вы можете реализовать их внутри класса адаптера или в вашем фрагменте или действии, но в этом случае вам нужно создать экземпляр адаптера для вызова методов уведомления. В моем случае я обычно реализую это в адаптере.
источник
Я обнаружил, что действительно простой способ перезагрузить RecyclerView - это просто вызвать
Это сначала удалит все содержимое RecyclerView, а затем добавит его снова с обновленными значениями.
источник
я получил ответ через долгое время
источник
Если ничего не упомянуто в вышеприведенных комментариях, это работает для вас. Это может означать, что проблема лежит где-то еще.
Единственное место, где я нашел решение, было в том, как я настраивал список для адаптера. В моей деятельности список представлял собой переменную экземпляра, и я менял ее напрямую, когда менялись какие-либо данные. Из-за того, что это ссылочная переменная, происходило что-то странное. Поэтому я изменил ссылочную переменную на локальную и использовал другую переменную для обновления данных, а затем перешел к
addAll()
функции, упомянутой в ответах выше.источник