Próbuję dowiedzieć się, na czym polega problem z aktualizacją RecyclerView
adaptera.
Po otrzymaniu nowej listy produktów próbowałem:
Zaktualizuj
ArrayList
z fragmentu, w którymrecyclerView
został utworzony, ustaw nowe dane na adapter, a następnie wywołajadapter.notifyDataSetChanged()
; to nie działało.Utwórz nowy adapter, tak jak inni, i działał on dla nich, ale dla mnie bez zmian:
recyclerView.setAdapter(new RecyclerViewAdapter(newArrayList))
Utwórz metodę, w
Adapter
której dane będą aktualizowane w następujący sposób:public void updateData(ArrayList<ViewModel> viewModels) { items.clear(); items.addAll(viewModels); notifyDataSetChanged(); }
Następnie wywołuję tę metodę za każdym razem, gdy chcę zaktualizować listę danych; to nie działało.
Aby sprawdzić, czy mogę w jakikolwiek sposób zmodyfikować recyklerView, próbowałem usunąć przynajmniej element:
public void removeItem(int position) { items.remove(position); notifyItemRemoved(position); }
Wszystko pozostało bez zmian.
Oto mój adapter:
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);
}
}
I inicjuję RecyclerView
w następujący sposób:
recyclerView = (RecyclerView) view.findViewById(R.id.recycler);
recyclerView.setLayoutManager(new GridLayoutManager(getActivity(), 5));
adapter = new RecyclerViewAdapter(items);
adapter.setOnItemClickListener(this);
recyclerView.setAdapter(adapter);
Jak więc zaktualizować dane adaptera, aby wyświetlać nowo otrzymane elementy?
Aktualizacja: problem polegał na tym, że wygląd gridView wyglądał następująco:
<?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>
Potem właśnie usunąłem LinearLayout
i utworzyłem FrameLayout
układ macierzysty.
źródło
items.clear(); items.addAll(newItems);
to brzydki wzór. Jeśli naprawdę potrzebujesz tutaj defensywnego kopiowania,items = new ArrayList(newItems);
byłoby mniej brzydkie.Odpowiedzi:
Pracuję z RecyclerView i zarówno usuwanie, jak i aktualizacja działają dobrze.
1) USUŃ: Istnieją 4 kroki, aby usunąć element z RecyclerView
Ta linia kodów działa dla mnie.
2) AKTUALIZACJA DANYCH: Jedyne, co musiałem zrobić, to
Musiałeś to wszystko zrobić w kodzie Actvity / Fragment, a nie w kodzie adaptera RecyclerView.
źródło
recycler.removeViewAt(position);
(a raczejrecycler.removeAllViews()
) były kluczowe.To ogólna odpowiedź dla przyszłych gości. Różne sposoby aktualizacji danych adaptera są wyjaśnione. Proces obejmuje dwa główne kroki za każdym razem:
Wstaw pojedynczy element
Dodaj „Pig” w indeksie
2
.Wstaw wiele elementów
Wstaw trzy kolejne zwierzęta według indeksu
2
.Usuń pojedynczy element
Usuń „Pig” z listy.
Usuń wiele elementów
Usuń „Wielbłąda” i „Owcę” z listy.
Usuń wszystkie elementy
Wyczyść całą listę.
Zastąp starą listę nową
Wyczyść starą listę, a następnie dodaj nową.
adapter
Ma odniesienie dodata
, więc ważne jest, aby nie ustawićdata
do nowego obiektu. Zamiast tego wyczyściłem stare elementy,data
a następnie dodałem nowe.Zaktualizuj pojedynczy element
Zmień element „Owca”, tak aby brzmiał „Lubię owce”.
Przenieś pojedynczy element
Przesuń „Owcę” z pozycji
3
na pozycję1
.Kod
Oto kod projektu w celach informacyjnych. Kod adaptera RecyclerView można znaleźć w tej odpowiedzi .
MainActivity.java
Notatki
notifyDataSetChanged()
, animacja nie zostanie wykonana. Może to być również kosztowna operacja, dlatego nie zaleca się korzystania z niej,notifyDataSetChanged()
jeśli aktualizujesz tylko jeden element lub szereg przedmiotów.Dalsze badanie
źródło
Update single item
, powinienem być jak żółwieOto, co zadziałało dla mnie:
Po utworzeniu nowego adaptera zawierającego zaktualizowaną listę (w moim przypadku była to baza danych przekonwertowana na ArrayList) i ustawieniu go jako adaptera, próbowałem
recyclerView.invalidate()
i działało.źródło
YourAdapter adapter = (YourAdapter) recyclerView.getAdapter(); adapter.yourSetterMethod(newList); adapter.notifyDataSetChanged();
To powiedziane, to brzmi jak to, co OP najpierw próbował (po prostu dodając to jako komentarz, aby mogło pomóc komuś innemu, jak to działało w moim przypadku).masz dwie opcje: odśwież interfejs użytkownika z adaptera:
lub odśwież go z recyklerViewView:
źródło
Inną opcją jest użycie diffutil . Porównuje oryginalną listę z nową listą i użyje nowej listy jako aktualizacji, jeśli nastąpi zmiana.
Zasadniczo możemy użyć DiffUtil, aby porównać stare dane z nowymi danymi i pozwolić mu na wywołanie powiadomienieIdentRangeRemoved, i powiadomienieItemRangeChanged i powiadomienieItemRangeInserted w Twoim imieniu.
Szybki przykład użycia diffUtil zamiast powiadomieniaDataSetChanged:
Obliczam różnicę od głównego wątku na wypadek, gdyby była to duża lista.
źródło
getItems()
. Skąd wiesz, które dane z oryginalnej listy należy usunąć / dodać / zaktualizować?Zaktualizuj dane widoku listy, widoku siatki i widoku recyklingu
lub
źródło
Najlepszym i najfajniejszym sposobem dodawania nowych danych do danych bieżących jest
źródło
Ten sam problem rozwiązałem w inny sposób. Nie mam danych, na które czekam z wątku w tle, więc zacznij od listy emty.
Otóż to.
źródło
Te metody są skuteczne i dobrze zacząć korzystać z podstawowych
RecyclerView
.Możesz zaimplementować je wewnątrz klasy adaptera lub we fragmencie lub działaniu, ale w takim przypadku musisz utworzyć instancję adaptera, aby wywołać metody powiadamiania. W moim przypadku zwykle implementuję go w adapterze.
źródło
Dowiedziałem się, że naprawdę prostym sposobem na ponowne załadowanie RecyclerView jest po prostu połączenie
To najpierw usunie całą zawartość RecyclerView, a następnie doda ją ponownie ze zaktualizowanymi wartościami.
źródło
dostałem odpowiedź po długim czasie
źródło
Jeśli nic nie wspomniane w powyższych komentarzach działa dla Ciebie. Może to oznaczać, że problem leży gdzie indziej.
Znalazłem jedno rozwiązanie, w którym ustawiłem listę adaptera. W mojej działalności lista była zmienną instancji i zmieniałem ją bezpośrednio po zmianie danych. Ponieważ była to zmienna referencyjna, działo się coś dziwnego. Zmieniłem więc zmienną referencyjną na lokalną i użyłem innej zmiennej do aktualizacji danych, a następnie przekazałem
addAll()
funkcję wymienioną w powyższych odpowiedziach.źródło