自從接觸了Rxjava之后就一直在使用Rxjava的方式來修改以前寫過的一些代碼,前段時間實(shí)現(xiàn)了Retrofit2+Rxjava的方式來請求數(shù)據(jù)感覺還不錯蚤告,那么既然使用了這種方式來獲取數(shù)據(jù)舔箭,自然也可以使用rx的方式來對數(shù)據(jù)進(jìn)行處理。
之前我們可能使用callback箫章、handler镜会、Otto戳表、EventBus...的方式處理數(shù)據(jù),現(xiàn)在我們就來看下怎樣自己制作簡單的rxbus實(shí)現(xiàn)對數(shù)據(jù)的處理镣屹。
備注:rxjava已經(jīng)升級為2.0下面是升級為2.0的RxBus傳送門
http://www.reibang.com/p/b22e6b10c6cf
1 使用方式
1.1 注冊與取消注冊
在合適的地方進(jìn)行注冊和取消注冊
RxBus.getInstance().register(this);
RxBus.getInstance().unRegister(this);
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
RxBus.getInstance().register(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
RxBus.getInstance().unRegister(this);
}
1.2 發(fā)布事件與訂閱事件
//發(fā)布
RxBus.getInstance().post(0, s.getBookses());
//訂閱
@Subscribe(tag = 0, thread = EventThread.MAIN_THREAD)
private void dataBinding(ArrayList<TravelNoteBook.Books> bookses) {
...//處理邏輯
}
當(dāng)數(shù)據(jù)被發(fā)布之后价涝,如果有訂閱者訂閱色瘩,就會獲取到發(fā)布過的數(shù)據(jù)。這也是一個最簡單的rxbus覆山,那么我們就來看一下rxbus背后的處理泥栖,后面的內(nèi)容最好對rxjava和lambda方面的知識有一些了解。
2 實(shí)現(xiàn)最簡單的rxbus
對于上面用注解方式來訂閱事件的方式,我們先用最簡單的方式實(shí)現(xiàn)晦毙,然后把注解方式推導(dǎo)出來
2.1 發(fā)布與訂閱
首先我們要有一個Subject 來處理發(fā)布和訂閱事件
private final Subject bus;
/**
* PublishSubject 創(chuàng)建一個可以在訂閱之后把數(shù)據(jù)傳輸給訂閱者Subject
* SerializedSubject 序列化Subject為線程安全的Subject
*/
public RxBus() {
bus = new SerializedSubject<>(PublishSubject.create());
}
然后我們來發(fā)布
public void post(int code, Object obj) {
bus.onNext(obj);
}
最后訂閱
public <T> Observable tObservable(final Class<T> eventType) {
return bus.ofType(eventType.getClass());//判斷接收事件類型
}
使用
APIServiceManager.getInstance()
.getTravelNotesAPI()
//Retrofit發(fā)送請求并返回數(shù)據(jù)
.getTravelNotesList(query, page + "")
//生命周期處理
.compose(bindToLifecycle())
//img01
.compose(RxSchedulersHelper.io_main())
.subscribe(s -> {
//發(fā)布
RxBus.getInstance().post(s.getBookses());
});
//訂閱
Subscription subscription=RxBus.getInstance().tObservable(POJO.class)
.subscribe(s -> {//事件處理});
//取消訂閱
subscription.unsubscribe();
現(xiàn)在最原始的rxbus就成功了,現(xiàn)在可以進(jìn)行簡單的事件的發(fā)布與訂閱甸陌,雖然可以進(jìn)行發(fā)布于訂閱盐股,但是在項目中還沒有辦法滿足我們的需求耻卡,那么下面我們就需要對現(xiàn)在的這種方式進(jìn)行優(yōu)化卵酪。
2.2 rxbus優(yōu)化
先看一下優(yōu)化后rxbus包的內(nèi)容
Subscribe //訂閱注解
EventTag //事件標(biāo)示
EventThread //事件線程
Msg //發(fā)布事件時默認(rèn)封裝pojo
RxBus //rxbus主要處理
一共五個類主要的就是rxbus溃卡,那我們就先看一下rxbus中到底做了哪些處理
2.2.1 注冊
然后需要一個map空間存放訂閱者方便取消注冊
//存放訂閱者信息
private Map<Object, List<Subscription>> subscriptions = new HashMap<>();
現(xiàn)在來處理注冊事件
下面代碼核心思想是,獲取Object 中被Subscribe注解的方法并通過反射的方式漩仙,在數(shù)據(jù)被發(fā)布后通過invoke調(diào)用方法犹赖,并在訂閱后把訂閱者加入map空間方便解除注冊。
public void register(Object subscriber) {
Observable.just(subscriber)
//判斷訂閱者不為空
.filter(s -> s != null)
.map(s -> s.getClass())
//獲取訂閱者方法并且用Observable裝載
.flatMap(s -> Observable.from(s.getDeclaredMethods()))
//方法必須被Subscribe注解
.filter(m -> m.isAnnotationPresent(Subscribe.class))
//使非public方法可以被invoke
.doOnNext(m -> m.setAccessible(true))
.subscribe(m -> {
addSubscription(m,subscriber);
});
}
private void addSubscription(Method m,Object subscriber){
//獲取方法內(nèi)參數(shù)
Class[] parameterType = m.getParameterTypes();
//只獲取第一個方法參數(shù)麸折,否則默認(rèn)為Object
Class cla = Object.class;
if (parameterType.length > 1) {
cla = parameterType[0];
}
//獲取注解
Subscribe sub = m.getAnnotation(Subscribe.class);
//訂閱事件
Subscription subscription = tObservable(sub.tag(), cla)
.observeOn(EventThread.getScheduler(sub.thread()))
.subscribe(o -> {
try {
m.invoke(subscriber, o);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}, e -> System.out.println("this object is not invoke"));
putSubscriptionsData(subscriber,subscription);
}
/**
* 添加訂閱者到map空間來unRegister
* @param subscriber 訂閱者
* @param subscription 訂閱者 Subscription
*/
private void putSubscriptionsData(Object subscriber,Subscription subscription){
List<Subscription> subs = subscriptions.get(subscriber);
if (subs == null) {
subs = new ArrayList<>();
}
subs.add(subscription);
subscriptions.put(subscriber, subs);
}
代碼中我們調(diào)用過這么一個方法:tObservable(sub.tag(), cla)
這個就是剛才的tObservable方法磕谅,然后增加了一個int類型的參數(shù)用于標(biāo)示在同一個Object下使用不同的事件處理方法
/**
* 訂閱事件
* @return
*/
public <T> Observable tObservable(int code, final Class<T> eventType) {
return bus.ofType(Msg.class)//判斷接收事件類型
.filter(new Func1<Msg, Boolean>() {
@Override
public Boolean call(Msg o) {
//過濾code相同的事件
return o.code == code;
}
})
.map(new Func1<Msg, Object>() {
@Override
public Object call(Msg o) {
return o.object;
}
})
.cast(eventType);
}
對于post也相應(yīng)的增加了int參數(shù)雾棺,通過對code的判斷bus會返回eventType類型的Observable數(shù)據(jù),并傳入備注解的相應(yīng)方法中衬浑。這樣就完成了整個事件的注冊過程
2.2.2 發(fā)布數(shù)據(jù)
把數(shù)據(jù)包裝以下并發(fā)送
public void post(int code, Object obj) {
bus.onNext(new Msg(code, obj));
}
2.2.3 取消注冊
public void unRegister(Object subscriber) {
if (subscriber == null) {
return;
}
List<Subscription> subs = subscriptions.get(subscriber);
if (subs != null) {
for (Subscription sub : subs) {
if (sub != null)
sub.unsubscribe();
}
subscriptions.remove(subscriber);
}
}
取消注冊主要就是判斷map中相同的subscriber捌浩,然后調(diào)用unsubscribe()取消事件訂閱并刪除相應(yīng)的map數(shù)據(jù),保證訂閱事件與activity生命周期相同防止內(nèi)存溢出工秩。
使用
APIServiceManager.getInstance()
.getTravelNotesAPI()
.getTravelNotesList(query, page + "")
.compose(bindToLifecycle())
.compose(RxSchedulersHelper.io_main())
//對返回數(shù)據(jù)的優(yōu)先處理尸饺,如果數(shù)據(jù)有異常則會執(zhí)行error
.compose(SchedulersHelper.handleResult())
.subscribe(s -> {
//發(fā)布
RxBus.getInstance().post(0, s.getBookses());
},
e -> {
//發(fā)布(error)
RxBus.getInstance().post(RxBus.TAG_ERROR, e.getMessage());
});
@Subscribe(tag = 0, thread = EventThread.MAIN_THREAD)
private void dataBinding(ArrayList<TravelNoteBook.Books> bookses) {
//事件處理
}
@Subscribe(tag =RxBus.TAG_ERROR)
private void dataError(String error) {
ToastUtil.getInstance().makeShortToast(this, error);
}
3 總結(jié)
現(xiàn)在優(yōu)化過后最簡單的rxbus事件總線就已經(jīng)實(shí)現(xiàn),可以對數(shù)據(jù)進(jìn)行發(fā)布和訂閱處理助币,相關(guān)的代碼和demo可以訪問我的git:
https://github.com/hackerlc/GearApplication
或者在gradle中直接引用:
compile 'com.joker.gear:com-joker-gear:1.4.1'
當(dāng)然因為現(xiàn)在rxbus是最簡單的版本浪听,不論是代碼還是實(shí)現(xiàn)方式還有很多不足的地方眉菱,在實(shí)際項目中也會遇到更多的問題迹栓,這里只提供了只用rxbus的實(shí)現(xiàn)思路,隨著rxbus的不斷完善相信后面會有更加健壯的代碼呈現(xiàn)出來俭缓。
4 相關(guān)引用
在實(shí)現(xiàn)rxbus上看了很多的文章和git克伊,也非常感謝這些作者的分享酥郭,當(dāng)然如果在文章引用過程中有什么問題,請聯(lián)系我我會作出修改愿吹,以下是相關(guān)的引用連接不从。
http://www.open-open.com/lib/view/open1444011371541.html
http://www.reibang.com/p/ca090f6e2fe2
https://github.com/AndroidKnife/RxBus
https://github.com/1030310877/JoeRxBus
https://github.com/ch331917692/RxBus
http://www.reibang.com/p/f3f0eccbcd6f