首页 Android 正文
  • 本文约2656字,阅读需13分钟
  • 128
  • 0

在Room中使用RxJava

摘要

更少的重复代码,编译时检查的SQL查询,除此之外还有异步功能和可观察的查询-听起来是不是很牛?有了 Room,这些都成为可能。异步查询返回LiveData或者RxJava的Maybe, Single或...

更少的重复代码,编译时检查的SQL查询,除此之外还有异步功能和可观察的查询-听起来是不是很牛?有了 Room,这些都成为可能。异步查询返回LiveData或者RxJavaMaybe, Single或者Flowable。它们都是可观察的查询,可以让你在在数据变更的时候自动获得更新,以便确保UI上反应的是数据库的最新数据。如果你已经在app中使用了RxJava 2 ,那么把Room和Maybe,Single以及Flowable一起使用应该会非常轻松。

让我们考虑下面的UI:用户可以查看和编辑他们的username。该数据以及其它的用户信息被保存在数据库里。

为了从数据库获得user,我们可以在UserDao中编写下面的查询:

@Query(“SELECT * FROM Users WHERE id = :userId”)
User getUserById(String userId);

这种方法有两个缺点:

  • 它是一个阻塞式的,同步的调用。
  • 每次数据变更的时候我们都需要手动调用一次这个方法。

Room提供了对数据库中数据进行观察的选项,并通过 RxJava的Maybe,SingleFlowable对象来执行异步查询。

要让Room支持RxJava 2,在build.gradle文件添加如下的dependencies:

// RxJava support for Room
implementation “android.arch.persistence.room:rxjava2:1.0.0-alpha5”
// Testing support
androidTestImplementation “android.arch.core:core-testing:1.0.0-alpha5”

Maybe

@Query(“SELECT * FROM Users WHERE id = :userId”)
Maybe<User> getUserById(String userId);
可能发生以下事情:
  • 当数据库中没有user,查询没有返回行时,Maybe调用complete。
  • 当数据库中有一个user时,Maybe将触发onSuccess并调用complete。
  • 如果在Maybe的complete调用之后user被更新,什么也不会发生。

Single

@Query(“SELECT * FROM Users WHERE id = :userId”)
Single<User> getUserById(String userId);
可能发生以下事情:
  • 当数据库中没有user,查询没有返回行时,Single触发onError(EmptyResultSetException.class)。
  • 当数据库中有一个user时, Single触发onSuccess。
  • 如果在Single.onComplete调用之后user被更新,什么也不会发生。

Flowable

@Query(“SELECT * FROM Users WHERE id = :userId”)
Flowable<User> getUserById(String userId);
可能发生以下事情:
  • 当数据库中没有user,查询没有返回行时,Flowable不会发射,也不会触发onNext, 或者 onError。
  • 当数据库中有一个user时,Flowable会触发onNext。
  • 每当user更新之后,Flowable将自动发射,这样你就可以根据最新的数据来更新UI。
译者注:这里的getUserById返回的是Flowable,如果查询没有结果的话,什么信息也收不到,这在实际开发中是很蛋疼的。但是如果我们把返回类型改成Flowable的话,如果查询没有结果是可以得到一个空list的。作者在回复中讲到之所以第一种情况没有通知是因为,在实现的过程中遇到了bug。不过我个人认为要解决这个问题应该不难吧。

测试

测试一个返回Maybe/Single/Flowable的query跟测试它相应的同步query没有什么大的区别。在UserDaoTest中,确保是用的是in-memory database,这样进程杀死之后存储的测试信息就自动被清理了。

@RunWith(AndroidJUnit4.class)
public class UserDaoTest {
…
private UsersDatabase mDatabase;
@Before
public void initDb() throws Exception {
    mDatabase = Room.inMemoryDatabaseBuilder(
                     InstrumentationRegistry.getContext(),
                     UsersDatabase.class)
            // allowing main thread queries, just for testing
            .allowMainThreadQueries()
            .build();
}
@After
public void closeDb() throws Exception {
    mDatabase.close();
}

在测试中添加InstantTaskExecutorRule,确保Room立即执行了所有的数据库操作。

@Rule
public InstantTaskExecutorRule instantTaskExecutorRule = new InstantTaskExecutorRule();

让我们实现一个订阅getUserById的test并在user被插入时检查得到的是否就是我们插入的数据。

@Test
public void insertAndGetUserById() {
    // Given that we have a user in the data source
    mDatabase.userDao().insertUser(USER);
    // When subscribing to the emissions of user
    mDatabase.userDao()
             .getUserById(USER.getId())
             .test()
             // assertValue asserts that there was only one emission
             .assertValue(new Predicate<User>() {
                @Override
                public boolean test(User user) throws Exception {
                    // The emitted user is the expected one
                    return user.getId().equals(USER.getId()) &&
                      user.getUserName().equals(USER.getUserName());
                }
            });
}
标签:RoomRxJava

扫描二维码,在手机上阅读


    评论