Java技巧分享之利用RxJava打造可觀測數(shù)據(jù)RxLiveData
1. 問題場景
在實際工作中,我們經(jīng)常需要在不同類對象之間、不同模塊之間共享數(shù)據(jù),而這些數(shù)據(jù)通常是可改動的,那么就可能發(fā)生一個問題:當數(shù)據(jù)變動時,相關(guān)對象或模塊并不知道,沒有及時更新數(shù)據(jù)。這時候,我們希望數(shù)據(jù)改變時可以通知其他模塊同步更新,實現(xiàn)一個類似數(shù)據(jù)之間聯(lián)動的效果。最容易想到的應該就是監(jiān)聽回調(diào)的觀察者模式,下面給出一種以前見過的、不太優(yōu)雅的實現(xiàn):
class User { //...... Java Bean 的字段略 } interface Listener { void onUserUpdated(User user); } class UserManager { private static UserManager manager = new UserManager(); private UserManager() { } public static UserManager getInstance() { return manager; } private User user; private List<Listener> listeners = new LinkedList<>(); public void addUserListener(Listener listener) { listeners.add(listener); } public void removeUserListener(Listener listener) { listeners.remove(listener); } public User getUser() { return user; } public void setUser(User user) { this.user = user; for (Listener listener : listeners) { listener.onUserUpdated(this.user); } } }
這種方式有以下缺點:
- 不具備復用性(每次添加新的數(shù)據(jù)都要把回調(diào)監(jiān)聽重新實現(xiàn)一遍);
- 增加內(nèi)存溢出的風險(調(diào)用
addUserListener
的人可能忘記調(diào)用removeUserListener
); - setter方法的污染(做了多余的事情)。
面對這樣的問題,RxJava、JDK中的Observable和Flow API還有Android里的LiveData都給出了可用的實現(xiàn)方式,在實際開發(fā)中,感覺并不是那么方便。而本文要介紹的是我利用RxJava打造一個更加方便的可觀測對象工具類--RxLiveData(代碼見最底部)。
2. 使用示例
先來看一個比較短的完整示例:
/* 測試用的 Java Bean 數(shù)據(jù)類*/ class User { //...... Java Bean 的字段略 } /* 一個單例 */ class UserManager { private static final UserManager manager = new UserManager(); private UserManager() { } public static UserManager getInstance() { return manager; } private final RxLiveData<User> userData = new RxLiveData<>(); public RxLiveData<User> getUserData() { return userData; } } class A { public void init() { //訂閱可觀測對象,使得數(shù)據(jù)發(fā)生改變時可以被回調(diào) UserManager.getInstance().getUserData().getObservable() .subscribe((User user) -> {//使用lambda版 update(user);// 每次用戶信息改變,這里會被調(diào)用 }); update(UserManager.getInstance().getUserData().getValue()); } private void update(User user) { System.out.println("user changed"); } } class B{ public B() { UserManager.getInstance().getUserData().getObservable().subscribe(this::update);//方法引用版 } private void update(User user) { System.out.println("user changed"); } } public class Main { public static void main(String[] args) throws InterruptedException { A a = new A(); a.init(); B b = new B(); //更新UserManager中的數(shù)據(jù),這時候A和B中的對應方法會被調(diào)用 UserManager.getInstance().getUserData().postData(new User()); } }
這里模擬UserManager
的數(shù)據(jù)在A
和B
類的對象之間共享,當UserManager
的內(nèi)容發(fā)生改變時,可以通知到A
和B
,執(zhí)行相應操作。
這時如果還想給UserManager
增加一個數(shù)據(jù),例如一個long
類型的time
,只需要按照下面這樣添加一個屬性和一個getter方法就可以了:
private final RxLiveData<Long> timeData = new RxLiveData<>(); public RxLiveData<Long> getTimeData() { return timeData; }
如果是在Android應用開發(fā)中,還可以借助RxAndroid和RxLifecycle的功能,來控制回調(diào)的執(zhí)行線程并在界面銷毀時取消訂閱,例如:
userManager.getUserData().getObservable() .compose(bindUntilEvent(ActivityEvent.DESTROY))//指定在onDestroy回調(diào)時取消訂閱 .observeOn(AndroidSchedulers.mainThread())//指定主線程 .subscribe(user -> { }, Throwable::printStackTrace);
3. 主要方法介紹
3.1 getObservable 方法
方法簽名:public Observable<T> getObservable()
這個方法用于獲取RxJava的Observable
,進而對數(shù)據(jù)進行訂閱,還可以得到RxJava相關(guān)功能的支持(例如,Stream 操作,指定線程,控制生命周期等等)。
3.2 postData 方法
方法簽名:public void postData(T value)
這個方法用于更新數(shù)據(jù)。它會更新存在在當前RxLiveData對象中的數(shù)據(jù),并通過RxJava的ObservableEmitter
觸發(fā)觀察者的回調(diào)。
注意:當參數(shù)為null
時,由于RxJava會對null
拋出異常,所以這里的實現(xiàn)方式是在判斷為null
的時候只存儲數(shù)據(jù),不觸發(fā)觀察者的回調(diào)。
3.3 getValue 方法
方法簽名:public T getValue()
這個方法僅用于獲得存在在當前RxLiveData中的數(shù)據(jù)。
3.4 optValue 方法
方法簽名:public Optional<T> optValue()
getValue
方法的Optional版本。
4. 完整實現(xiàn)
import io.reactivex.rxjava3.core.Observable;//如果用的是RxJava2的請改為該版本的包名 import io.reactivex.rxjava3.core.ObservableEmitter; import io.reactivex.rxjava3.disposables.Disposable; import java.util.Optional; public class RxLiveData<T> { private final Observable<T> observable; private Disposable disposable; private T value; private ObservableEmitter<T> emitter; public RxLiveData() { observable = Observable .create((ObservableEmitter<T> emitter) -> this.emitter = emitter) .publish() .autoConnect(0, disposable -> this.disposable = disposable); } public Observable<T> getObservable() { return observable; } public void postData(T value) { this.value = value; if (emitter != null && value != null) { emitter.onNext(value); } } public T getValue() { return value; } public Optional<T> optValue() { return Optional.ofNullable(value); } }
到此這篇關(guān)于Java技巧分享之利用RxJava打造可觀測數(shù)據(jù)RxLiveData的文章就介紹到這了,更多相關(guān)Java RxJava可觀測數(shù)據(jù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
基于SpringBoot實現(xiàn)驗證碼功能的代碼及思路
SpringBoot技術(shù)是目前市面上從事JavaEE企業(yè)級開發(fā)過程中使用量最大的技術(shù),下面這篇文章主要給大家介紹了如何基于SpringBoot實現(xiàn)驗證碼功能的相關(guān)資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下2024-07-07前端如何調(diào)用后端接口進行數(shù)據(jù)交互詳解(axios和SpringBoot)
一般來講前端不會給后端接口,而是后端給前端接口的情況比較普遍,下面這篇文章主要給大家介紹了關(guān)于前端如何調(diào)用后端接口進行數(shù)據(jù)交互的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-03-03關(guān)于解決雪花算法生成的ID傳輸前端后精度丟失問題
這篇文章主要介紹了關(guān)于解決雪花算法生成的ID傳輸前端后精度丟失問題,雪花算法生成的ID傳輸?shù)角岸藭r,會出現(xiàn)后三位精度丟失,本文提供了解決思路,需要的朋友可以參考下2023-03-03