#java #android #android-recyclerview #pagedlist
#java #Android #android-recyclerview #список страниц
Вопрос:
Я только что внедрил новую архитектуру компонентов Android, Room, ViewModel, LiveData, Retrofit и RecyclerView, так что все в порядке, проблема в том, что список, который я показываю в recyclerview, экспоненциально велик, поскольку это прототип для чата, и приложение может немного зависнуть.
При поиске в Интернете я нашел документацию об адаптере PagedList, и это может быть возможным решением, проблема в том, что в MainActivity, когда я вызываю observer, куда я должен поместить adapter.submitList(сообщение), (внутри функции onchange), я не показываю данные
MainActivity.class
public class MainActivity extends AppCompatActivity {
public static final int NEW_WORD_ACTIVITY_REQUEST_CODE = 1;
private MenViewModel mWordViewModel;
MensajesAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
RecyclerView recyclerView = findViewById(R.id.recyclerview);
adapter = new MensajesAdapter( getApplicationContext());
mWordViewModel =
ViewModelProviders.of(this).get(MenViewModel.class);
recyclerView.setHasFixedSize(true);
mWordViewModel.usersList.observe(this, mensajes -> {
adapter.submitList(mensajes);
});
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.setAdapter(adapter);
Странно то, что если я изменю код submitList с помощью adapter.setwords (параметр), список, если он отображается, но, конечно, это будет использовать notifydatasetchange (), и это будет неэффективно
PagedListAdapter—>MensajesAdapter.class
import android.arch.paging.PagedListAdapter;
import android.content.Context;
import android.support.annotation.NonNull;
import android.support.v7.util.DiffUtil;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import java.util.List;
public class MensajesAdapter extends PagedListAdapter<Mensajes,MensajesAdapter.MensajesViewHolder> {
private Context mCtx;
private List<Mensajes> heroList;
MensajesAdapter( Context context) {
super(DIFF_CALLBACK);
this.mCtx=context;
}
**void setWords(List<Mensajes> words){
heroList = words;
notifyDataSetChanged();
}**
@NonNull
@Override
public MensajesViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from(mCtx).inflate(R.layout.recyclerview_layout, parent, false);
return new MensajesViewHolder(view);
}
@Override
public void onBindViewHolder(@NonNull MensajesViewHolder mensajesViewHolder, int i) {
Mensajes men =getItem(i);
if(men!=null) {
mensajesViewHolder.textView.setText(men.getMensaje());
}else{
mensajesViewHolder.clear();
}
}
@Override
public int getItemCount() {
if (heroList != null)
return heroList.size();
else return 0;
}
public static DiffUtil.ItemCallback<Mensajes> DIFF_CALLBACK = new DiffUtil.ItemCallback<Mensajes>() {
@Override
public boolean areItemsTheSame(@NonNull Mensajes oldItem, @NonNull Mensajes newItem) {
Log.e("areITems","" (oldItem.getId() == newItem.getId()));
return oldItem.getId() == newItem.getId();
}
@Override
public boolean areContentsTheSame(@NonNull Mensajes oldItem, @NonNull Mensajes newItem) {
Log.e("areITems","" (oldItem.getMensaje().equals(newItem.getMensaje())));
return oldItem.getMensaje().equals(newItem.getMensaje());
}
};
class MensajesViewHolder extends RecyclerView.ViewHolder {
TextView textView;
public MensajesViewHolder(View itemView) {
super(itemView);
textView = itemView.findViewById(R.id.mensaje);
}
void clear()
{
textView.invalidate();
}
}
}
ViewModel—>MenViewModel.class
public class MenViewModel extends AndroidViewModel {
public MensajesDao countryDao;
private MenRepository mRepository;
// Using LiveData and caching what getAlphabetizedWords returns has several benefits:
// - We can put an observer on the data (instead of polling for changes) and only update the
// the UI when the data actually changes.
// - Repository is completely separated from the UI through the ViewModel.
public LiveData<PagedList<Mensajes>> mAllWords;
public LivePagedListBuilder livePlyerList;
public final LiveData<PagedList<Mensajes>> usersList;
public MenViewModel(Application application) {
super(application);
mRepository = new MenRepository(application);
mAllWords = mRepository.getAllWords();
MenRoomDatabase countryDatabase = MenRoomDatabase.getDatabase(getApplication());
countryDao = countryDatabase.mensajeDao();
//mRepository.recargar();
usersList =new LivePagedListBuilder<>(
countryDao.getAllMensajes(), /* page size */ 20).build();
livePlyerList=new LivePagedListBuilder<>(countryDatabase.mensajeDao().getAllMensajes(),new PagedList.Config.Builder()
.setPageSize(20)
.setPrefetchDistance(10)
.setPageSize(20)
.build());
}
LiveData<PagedList<Mensajes>> getAllWords() { return mAllWords; }
public void insert(Mensajes mensaje) { mRepository.insert(mensaje); }
Кто-нибудь знает, чтобы сказать мне, что не так? Чего мне не хватает или что у меня есть?
Комментарии:
1. Можете ли вы показать нам, каким образом вы использовали RecyclerView и как вы адаптировали изменения для миграции в PagedListAdapter?
2. Адаптер устанавливается в Recyclerview с пустыми данными, которые затем вызываются
submitList
в наблюдателе. Это означает, что Recyclerview не знает, что данные обновлены, что является причиной, по которой данные не отображаются. Согласно <a rel="nofollow noreferrer noopener" href="https://developer.android.com/reference/android/arch/paging/PagedListAdapter.html#submitList(android.arch.paging.PagedList)» rel=»nofollow noreferrer»> developer.android.com/reference/android/arch/paging / … , адаптер сохраняет данные перед назначением Recyclerview. В вашем случае, я думаю, вам нужно установить Adapter после загрузки данных в адаптер. Или вызовите notifidatasetchange.3. Большое вам спасибо, проблема была решена путем изменения @Override public int getItemCount () { вернуть super.getItemCount (); }
Ответ №1:
Убедитесь, что countryDao.getAllMensajes()
это обернуто с помощью DataSource.Factory<Int, Mensajes>
, и в вашем адаптере вам не следует управлять списком вручную, поскольку вы используете PagedListAdapter
.
В вашем @Override public int getItemCount() {}
do:
@Override
public int getItemCount() {
return super.getItemCount()
}
Комментарии:
1. БОЖЕ, я был на грани смены карьеры! Спасибо!