gpt4 book ai didi

android - 从数据库+网络加载数据(Room + Retrofit + RxJava2)

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:34:12 24 4
gpt4 key购买 nike

我有一个示例 API 请求,它返回用户关注列表的列表。我想在用户加载监视列表屏幕时实现以下流程:

  1. 立即从数据库缓存中加载数据。(cacheWatchList)

  2. 在后台启动 RetroFit 网络调用。

    我。 onSuccess 返回 apiWatchList
    二. onError 返回 cacheWatchList

  3. cacheWatchListapiWatchList 的区别

    我。相同 -> 一切正常,因为数据已经显示给用户,什么都不做。

    二。不同 -> 将 apiWatchList 保存到本地存储并将 apiWatchList 发送到下游。

到目前为止我做了什么?

Watchlist.kt

data class Watchlist(
val items: List<Repository> = emptyList()
)

LocalStore.kt(安卓机房)

  fun saveUserWatchlist(repositories: List<Repository>): Completable {    
return Completable.fromCallable {
watchlistDao.saveAllUserWatchlist(*repositories.toTypedArray())
}
}

RemoteStore.kt(改造 api 调用)

  fun getWatchlist(userId: UUID): Single<Watchlist?> {
return api.getWatchlist(userId)
}

DataManager.kt

  fun getWatchlist(userId: UUID): Flowable<List<Repository>?> {
val localSource: Single<List<Repository>?> =
localStore.getUserWatchlist()
.subscribeOn(scheduler.computation)

val remoteSource: Single<List<Repository>> = remoteStore.getWatchlist(userId)
.map(Watchlist::items)
.doOnSuccess { items: List<Repository> ->
localStore.saveUserWatchlist(items)
.subscribeOn(scheduler.io)
.subscribe()
}
.onErrorResumeNext { throwable ->
if (throwable is IOException) {
return@onErrorResumeNext localStore.getUserWatchlist()
}
return@onErrorResumeNext Single.error(throwable)
}
.subscribeOn(scheduler.io)

return Single.concat(localSource, remoteSource)
}

上述流程的问题是,它为下游(呈现器)的每个流源调用 onNext 两次,即使两个数据相同。

我可以在演示器中执行数据差异逻辑并进行相应更新,但我希望 DataManager 类为我处理逻辑(CleanArchitecture,SOC)。

我的问题?

  1. 实现上述逻辑的最佳方式是什么?

  2. 我是否泄露了 DataManager 中的内部订阅(参见:doOnSuccess 代码)?当演示者被销毁时,我将处理外部订阅。

最佳答案

fun getWatchlist(userId: UUID): Observable<List<Repository>?> {
val remoteSource: Single<List<Repository>> =
remoteStore.getWatchlist(userId)
.map(Watchlist::items)
.subscribeOn(scheduler.io)

return localStore.getUserWatchlist()
.flatMapObservable { listFromLocal: List<Repository> ->
remoteSource
.observeOn(scheduler.computation)
.toObservable()
.filter { apiWatchList: List<Repository> ->
apiWatchList != listFromLocal
}
.flatMapSingle { apiWatchList ->
localSource.saveUserWatchlist(apiWatchList)
.andThen(Single.just(apiWatchList))
}
.startWith(listFromLocal)
}
}

逐步解释:

  1. 从本地存储加载数据
  2. 每次 localStore 发出数据时,使用 flatMapObservable 订阅 remoteSource。
  3. 由于内部 Observable 有不止一个发射(来自本地的初始数据和来自远程源的更新数据的新数据)将 Single 转换为 Observable。
  4. 将来自 remoteSource 的数据与来自 localStore 的数据进行比较,并仅在 newData != localData 的情况下才处理数据。
  5. 对于过滤器启动 localSource 后的每个发射来保存数据,并在完成此操作后继续将数据保存为 Single。
  6. 根据要求,在远程请求开始时应该处理来自 localStore 的数据,只需在运算符链的末尾添加 startWith 即可。

关于android - 从数据库+网络加载数据(Room + Retrofit + RxJava2),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47367632/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com