- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在构建一个应用程序,该应用程序使用 ArticleBoundaryCallback 来初始化对 API 的调用,并将响应存储在 Room 中。我也在使用 LiveData 收听该表,并在 PagedListAdapter 中显示项目。
问题是每次在 Room 的(文章)表中插入新数据时,整个列表都会刷新。
此外,在配置更改时,似乎再次获取了整个数据(ViewModel 不保留它,RecyclerView 被重新创建)。
在每次插入时, RecyclerView 都会跳转(如果插入的是新数据,则跳转几行,或者如果它用旧数据替换新数据,则在开头)。
整个代码在这个 GitHub repo .
我的类(class)是:
文章:
@Entity(tableName = "article",
indices={@Index(value="id")})public class Article {
@PrimaryKey(autoGenerate = false)
@SerializedName("_id")
@Expose
@NonNull
private String id;
@SerializedName("web_url")
@Expose
private String webUrl;
@SerializedName("snippet")
@Expose
private String snippet;
@SerializedName("print_page")
@Expose
private String printPage;
@SerializedName("source")
@Expose
private String source;
@SerializedName("multimedia")
@Expose
@Ignore
private List<Multimedium> multimedia = null;
@Dao
public interface ArticleDao {
@Insert(onConflict = OnConflictStrategy.REPLACE)
long insert(Article article);
@Insert(onConflict = OnConflictStrategy.REPLACE)
void update(Article... repos);
@Insert(onConflict = OnConflictStrategy.REPLACE)
void insertArticles(List<Article> articles);
@Delete
void delete(Article... articles);
@Query("DELETE FROM article")
void deleteAll();
@Query("SELECT * FROM article")
List<Article> getArticles();
@Query("SELECT * FROM article")
DataSource.Factory<Integer, Article> getAllArticles();}
public class LocalCache {
private static final String TAG = LocalCache.class.getSimpleName();
private ArticleDao articleDao;
private Executor ioExecutor;
public LocalCache(AppDatabase appDatabase, Executor ioExecutor) {
this.articleDao = appDatabase.getArticleDao();
this.ioExecutor = ioExecutor;
}
public void insertAllArticles(List<Article> articleArrayList){
ioExecutor.execute(new Runnable() {
@Override
public void run() {
Log.d(TAG, "inserting " + articleArrayList.size() + " repos");
articleDao.insertArticles(articleArrayList);
}
});
}
public void otherFunction(ArrayList<Article> articleArrayList){
// TODO
}
public DataSource.Factory<Integer, Article> getAllArticles() {
return articleDao.getAllArticles();
}
public class AppRepository {
private static final String TAG = AppRepository.class.getSimpleName();
private static final int DATABASE_PAGE_SIZE = 20;
private Service service;
private LocalCache localCache;
private LiveData<PagedList<Article>> mPagedListLiveData;
public AppRepository(Service service, LocalCache localCache) {
this.service = service;
this.localCache = localCache;
}
/**
* Search - match the query.
*/
public ApiSearchResultObject search(String q){
Log.d(TAG, "New query: " + q);
// Get data source factory from the local cache
DataSource.Factory dataSourceFactory = localCache.getAllArticles();
// every new query creates a new BoundaryCallback
// The BoundaryCallback will observe when the user reaches to the edges of
// the list and update the database with extra data
ArticleBoundaryCallback boundaryCallback = new ArticleBoundaryCallback(q, service, localCache);
// Get the paged list
LiveData data = new LivePagedListBuilder(dataSourceFactory, DATABASE_PAGE_SIZE)
.setBoundaryCallback(boundaryCallback)
.build();
mPagedListLiveData = data;
ApiSearchResultObject apiSearchResultObject = new ApiSearchResultObject();
apiSearchResultObject.setArticles(data);
return apiSearchResultObject;
}
public DataSource.Factory getAllArticles() {
return localCache.getAllArticles();
}
public void insertAllArticles(ArrayList<Article> articleList) {
localCache.insertAllArticles(articleList);
}
}
public class DBArticleListViewModel extends ViewModel {
private AppRepository repository;
// init a mutable live data to listen for queries
private MutableLiveData<String> queryLiveData = new MutableLiveData();
// make the search after each new search item is posted with (searchRepo) using "map"
private LiveData<ApiSearchResultObject> repositoryResult = Transformations.map(queryLiveData, queryString -> {
return repository.search(queryString);
});
// constructor, init repo
public DBArticleListViewModel(@NonNull AppRepository repository) {
this.repository = repository;
}
// get my Articles!!
public LiveData<PagedList<Article>> articlesLiveData = Transformations.switchMap(repositoryResult, object ->
object.getArticles());
// get teh Network errors!
public LiveData<String> errorsLiveData = Transformations.switchMap(repositoryResult, object ->
object.getNetworkErrors());
// Search REPO
public final void searchRepo(@NonNull String queryString) {
this.queryLiveData.postValue(queryString);
}
// LAST Query string used
public final String lastQueryValue() {
return (String)this.queryLiveData.getValue();
}
DummyPagedListAdapter articleListAdapter = new DummyPagedListAdapter(this);
localDBViewModel = ViewModelProviders.of(this, Injection.provideViewModelFactory(this)).get(DBArticleListViewModel.class);
localDBViewModel.articlesLiveData.observe(this, pagedListLiveData ->{
Log.d(TAG, "articlesLiveData.observe size: " + pagedListLiveData.size());
if(pagedListLiveData != null)
articleListAdapter.submitList(pagedListLiveData);
});
recyclerView.setAdapter(articleListAdapter);
public class DummyPagedListAdapter extends PagedListAdapter<Article, ArticleViewHolder> {
private final ArticleListActivity mParentActivity;
public DummyPagedListAdapter(ArticleListActivity parentActivity) {
super(Article.DIFF_CALLBACK);
mParentActivity = parentActivity;
}
@NonNull
@Override
public ArticleViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(mParentActivity).inflate(R.layout.article_list_content, parent, false);
return new ArticleViewHolder(itemView);
}
@Override
public void onBindViewHolder(@NonNull ArticleViewHolder holder, int position) {
Article article = getItem(position);
if (article != null) {
holder.bindTo(article);
} else {
holder.clear();
}
}
}
public static DiffUtil.ItemCallback<Article> DIFF_CALLBACK = new
DiffUtil.ItemCallback<Article>() {
@Override
public boolean areItemsTheSame(@NonNull Article oldItem, @NonNull
Article newItem) {
return oldItem.getId() == newItem.getId();
}
@Override
public boolean areContentsTheSame(@NonNull Article oldItem, @NonNull
Article newItem) {
return oldItem.getWebUrl() == newItem.getWebUrl();
}
};
最佳答案
是的..花了我一段时间但我解决了它。正如我所想的,愚蠢的问题:在适配器用来决定从观察到的数据集中添加什么和忽略什么的 DIFF_CALLBACK 中,我使用作为比较器 oldItem.getId() == newItem.getId() 这是字符串! !!当然,适配器总是获得“新值”并添加它们。
更正 DiffUtil.ItemCallback
public static DiffUtil.ItemCallback<Article> DIFF_CALLBACK = new DiffUtil.ItemCallback<Article>() {
@Override
public boolean areItemsTheSame(@NonNull Article oldItem, @NonNull Article newItem) {
return oldItem.getStoreOrder() == newItem.getStoreOrder();
}
@Override
public boolean areContentsTheSame(@NonNull Article oldItem, @NonNull Article newItem) {
return oldItem.getId().equals(newItem.getId()) && oldItem.getWebUrl().equals(newItem.getWebUrl());
}
};
关于android-viewmodel - 从 Room 观察 LiveData 会导致 PagedListAdapter 刷新,ViewModel 也会在方向更改时刷新数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51455071/
我正在试用 Realm 以及包括 LiveData 在内的 Android 架构组件。 我一直在关注 Google 的应用程序架构指南: https://developer.android.com/t
我有两个 DAO、两个存储库和两个 POJO。有什么方法可以创建两个的一个 Livedata?我需要它来为 Recyclerview 制作单个列表。POJO 是类似的对象。 费用库: public c
使用PagedList,这里没有数据库备份,而是内存中的数据列表(称其为CachedDataList),可以通过fetchMore()填充功能。 有了 PositionalDataSource、Dat
我正在创建具有MVVM架构的应用程序,但遇到了一个问题:获取要在View中显示的LiveData列表。 在我的ViewModel中,我有一个getAll()函数,该函数使用Room从数据库中检索字符串
我有一个包含用户列表的 LiveData 对象,我正在尝试将数据传输到另一个 LiveData 对象以在其他地方使用。 我在 Room 中使用 MVVM,所以我从数据库和 ViewModel 中获取
我是测试新手,我想学习如何使用 MVVM 模式测试协程。我刚刚关注了 https://github.com/android/architecture-samples 项目并做了一些更改(删除了远程源)
我在存储库类中有一个方法如下,它从本地数据库或网络返回国家列表作为 LiveData: fun loadCountries(): LiveData>> { return object : Ne
official Android developer docs 中的协程 LiveData 示例使用 emit() 给出以下示例: val user: LiveData = liveData {
在文字游戏应用中,我在 Activity 和 fragment 之间共享一个模型: public class MainViewModel extends AndroidViewModel {
我正在尝试对一个 View 进行数据绑定(bind),该 View 应该显示通过 View 模型中的 LiveData 属性公开的数据,但我发现无法将 LiveData 内的对象绑定(bind)到 V
使用 lifecycle-viewmodel-ktx和 lifecycle-livedata-ktx并给出以下示例: ViewModel 实现: class AutocompletionViewMod
我正在尝试合并 Android Architecture GitHub example与数据绑定(bind)。为此,我想我必须在 UserViewModel 中添加一个从 LiveData> 到 Li
我需要将域对象映射到 UI 对象并使用实时分页列表显示。 我试图映射 LiveData>至 LiveData>和 map PositionalDataSource至 PositionalDataSou
假设我们有两个 LiveData 对象: LiveData> fooList; LiveData> barList; 并且可以通过某种方法(或构造函数)将 Foo 转换为 Bar 对象。将具有 Foo
我想使用 MutableLiveData 观察来自 ViewModel 的一些数据。问题是我使用子类和父类,并且与 LiveData 存在一些不兼容。我想在 Kotlin 中做的事情的一个例子: im
在 Java 中,我们访问 Livedata> 的第一个元素书籍使用 books[0] ,如何在 Kotlin 中做到这一点? 我已经尝试了一切。 //Java代码 books[0]; // Kotl
我阅读了许多关于 viewmodel 和 livedata 的教程,但我并没有在 mvvm 模式的 viewmodel 类中实际使用 livedata。提前致谢。 最佳答案 Android 开发人员通
我正在创建一个 TODO 列表应用程序,用户可以在其中添加要执行的任务。任务有几类。我有以下情况: viewModelMainActivity = new ViewModelProvi
我正在使用 Kotlin 和 Android 架构组件(LiveData 和 Room)制作一个 Android 应用。 我有一个显示用户的 Activity (存储在数据库中并使用带有 ViewMo
我在我的存储库中使用 AsyncTask,它用于设置存储库中的 LiveData。我如何使用转换从我的 ViewModel 观察这个 LiveData? 最佳答案 你可以忽略我的另一个答案。解决方案是
我是一名优秀的程序员,十分优秀!