使用 RxJava 从几个数据源中加载数据
假设我需要从网络上获取一些数据。每次需要数据的时候,我都可以简单的访问网络,但是,将数据缓存到磁盘或内存则可以更有效率。
更明确的说,我希望是这样的:
- 偶尔的从网络上获取新数据。
- 然而可以尽快的恢复数据(通过缓存网络数据的结果)。
我想要使用 RxJava 实现。
基本模式
给每一数据源(网络、磁盘和内存)一个 Observable<Data>
接口,我们可以通过两个操作: concat()
和 first()
,来实现一个简单的解决方案。
让我们看看它是如何实现的:
```java
// Our sources (left as an exercise for the reader)
Observable<Data> memory = ...;
Observable<Data> disk = ...;
Observable<Data> network = ...;
// Retrieve the first source with data
Observable<Data> source = Observable
.concat(memory, disk, network)
.first();
这种模式的关键是 concat()
只在需要资源的时候才会订阅每个子 Observable
。如果数据被缓存,就不需要通过速度慢的数据源来获取数据。
注意 concat()
中 Observables
数据源的顺序问题,因为它们是被一个接一个检索出来的。
保存数据
很显然,下一步就是保存数据源。如果,你没有将网络请求的结果保存到磁盘,将磁盘的地址保存在内存中,那就再也没法挽救啦!上面所有的代码就是让网咯请求持久化。
我的解决方式是在每次发出请求的时候保存或缓存数据源:
Observable<Data> networkWithSave = network.doOnNext(data -> {
saveToDisk(data);
cacheInMemory(data);
});
Observable<Data> diskWithCache = disk.doOnNext(data -> {
cacheInMemory(data);
});
现在,如果你使用 networkWithSave
和 diskWithCache
,数据都将会在你下载的时候自动保存。
(这种策略的另外一个好处就是 networkWithSave/diskWithCache
可以在任何地方使用,不仅仅在我们的多个数据源模式下。)
旧数据
很不幸,现在我们保存数据的代码运行的太好啦!它总是会返回相同的数据,不管数据有没有过期。记着,我想要返回到偶尔从服务器获取新的数据。
这个解决方案在 first()
中,我们可以执行筛选,只需要过滤掉那些过期的数据:
Observable<Data> source = Observable
.concat(memory, diskWithCache, networkWithSave)
.first(data -> data.isUpToDate());
现在,我们只会获取被证明是最新的条目。因此,如果我们的资源有一条旧数据,我们就会略过,直接到下一条数据,直到我们找到新数据。
first()vs.takeFirst()
你可以选择使用 first()
,你也可以选择使用 takeFirst()
。
这两种调用的区别是当没有可用资源的时候, first()
会抛出一个 NoSuchElementException
的异常,然而, takeFirst()
只会简单的结束。
你使用哪种就看你是否需要明确的处理数据是否用完。
代码案例
这里是以上代码的运用,你可以在这儿 checkout 出代码: https://github.com/dlew/rxjava-multiple-sources-sample
如果你需要一个实际案例,checkout 出 Gfycat 的应用,它是使用这种模式 查看 Gfycat 中的动态数据 。Gfycat 中的代码并没有使用以上的所有功能(因为它不需要),但是它演示了 concat().first()
的基本用法。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论