情况是这样的:
我有域层为业务逻辑提供数据获取接口,我有 2 个数据源:本地数据库和远程网络。
它是这样工作的:
- 请求所有用户:
DataRepository.getInstance().getUsers();
- 在 DataRepository 中,有 2 个来源:
LocalDataSource.getUsers()
它从本地数据库中获取所有用户,如果没有数据则忽略此请求。RemoteDataSource.getUsers()
它从我们的服务器请求最新的用户列表(即使本地数据库中有数据,为了保持数据更新),当请求数据时,将其保存或更新到本地数据库并发回结果。
我知道我可以通过以下方式实现我的目标DataRepository
:
public Observable<List<User>> getUsers() {
return Observable.create(new Observable.OnSubscribe<List<User>>() {
@Override
public void call(Subscriber<? super List<User>> subscriber) {
// 1. Request users from local database
List<User> localUsers = mLocalDataSource.getUsers();
if (!localUsers.isEmpty()) {
subscriber.onNext(localUsers);
}
// 2. Request the latest user list from server
// Send a retrofit2 request
Call<List<User>> call = mRemoteDataSource.getUsers();
try {
List<User> networkUsers = call.execute().body();
mLocalDataSource.saveUsers(networkUsers);
subscriber.onNext(networkUsers);
subscriber.onCompleted();
} catch (IOException e) {
subscriber.onError(e);
}
}
});
}
现在想想我已经在项目中使用了 rxjava,为什么不使用 SqlBrite 和 Retrofit2 RxAdapters 来做这件事更方便呢?所以LocalDataSource.getUsers()
现在返回Observable<List<User>>
,所以返回RemoteDataSource.getUsers()
。
本地数据源.java
public Observable<User> getUsers() {
final String sqlQuery = String.format("SELECT * FROM %s", UserTable.TABLE_NAME);
return mDatabaseHelper.createQuery(UserTable.TABLE_NAME, sqlQuery)
.mapToList(new Func1<Cursor, User>() {
@Override
public User call(Cursor c) {
return UserTable.parseCursor(c);
}
});
}
远程数据源.java
public Observable<List<User>> getUsers() {
return mRetrofitApi.users();
}
问题:
我应该怎么做DataRepository.getUsers()
才能实现与旧技巧相同的效果?
public Observable<List<User>> getUsers() {
Observable<List<User>> localUsers = mLocalDataSource.getUsers();
Observable<List<User>> remoteUsers = mRemoteDataSource.getUsers()
.flatMap(new Func1<List<User>, Observable<User>>() {
@Override
public Observable<User> call(List<User> users) {
return Observable.from(users);
}
})
.doOnNext(new Action1<User>() {
@Override
public void call(User user) {
mLocalDataSource.saveUser(user);
}
})
.toList();
// What should I return to make two observables both able to emit results to the Subscriber
return Observable.concat(localUsers, remoteUsers); // ???
}