在Room中使用RxJava

原文:Room and RxJava。 

ezgif-2-8463107ab5.jpg

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

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

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

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

这种方法有两个缺点:

  1. 它是一个阻塞式的,同步的调用。

  2. 每次数据变更的时候我们都需要手动调用一次这个方法。

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

要让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);

可能发生以下事情:

  1. 当数据库中没有user,查询没有返回行时,Maybe调用complete。

  2. 当数据库中有一个user时,Maybe将触发onSuccess并调用complete。

  3. 如果在Maybe的complete调用之后user被更新,什么也不会发生。

Single

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

可能发生以下事情:

  1. 当数据库中没有user,查询没有返回行时,Single触发onError(EmptyResultSetException.class)。

  2. 当数据库中有一个user时, Single触发onSuccess。

  3. 如果在Single.onComplete调用之后user被更新,什么也不会发生。

Flowable

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

可能发生以下事情:

  1. 当数据库中没有user,查询没有返回行时,Flowable不会发射,也不会触发onNext, 或者 onError。

  2. 当数据库中有一个user时,Flowable会触发onNext。

  3. 每当user更新之后,Flowable将自动发射,这样你就可以根据最新的数据来更新UI。

译者注:这里的getUserById返回的是Flowable,如果查询没有结果的话,什么信息也收不到,这在实际开发中是很蛋疼的。但是如果我们把返回类型改成Flowable<List>的话,如果查询没有结果是可以得到一个空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());
                }
            });
}

好了,以上就是本文的全部。你可以在这里查看使用 Room 和 RxJava的 sample app。

来自:官方ORM框架Room