Handler
為什么要使用Handler骗奖?因?yàn)槿绻麑⒑臅r(shí)操作放在主線程中會(huì)導(dǎo)致ANR嫩舟,這個(gè)時(shí)候我們需要把這些耗時(shí)的操作爆安,放在一個(gè)子線程中,因?yàn)樽泳€程涉及到UI更新叛复,但是當(dāng)子線程中有涉及到操作UI的操作時(shí),就會(huì)對(duì)主線程產(chǎn)生危險(xiǎn)扔仓,也就是說(shuō)褐奥,更新UI只能在主線程中更新,在子線程中操作是危險(xiǎn)的.
這個(gè)時(shí)候翘簇,Handler就出現(xiàn)了來(lái)解決這個(gè)復(fù)雜的問(wèn)題撬码,由于Handler運(yùn)行在主線程中(UI線程中),它與子線程可以通過(guò)Message對(duì)象來(lái)傳遞數(shù)據(jù)版保,這個(gè)時(shí)候呜笑,Handler就承擔(dān)著接受子線程傳過(guò)來(lái)的(子線程用sedMessage()方法傳遞)Message對(duì)象,(里面包含數(shù)據(jù)), 把這些消息放入主線程隊(duì)列中彻犁,配合主線程進(jìn)行更新UI叫胁。
對(duì)于發(fā)送消息的handler和處理消息的handler需要進(jìn)行傳遞進(jìn)行關(guān)聯(lián),需要在View層例如Activity或者Fragment里需要定義handler并通過(guò)handleMessage()方法對(duì)傳遞的msg進(jìn)行處理汞幢;對(duì)于實(shí)際發(fā)送消息的VM層或者P層的類需要又View層將handler傳遞過(guò)去才能進(jìn)行處理并在handler中處理驼鹅,這樣就是的View層和VM層存在耦合;
EventBus
EventBus是一個(gè)Android端優(yōu)化的publish/subscribe消息總線森篷,簡(jiǎn)化了應(yīng)用程序內(nèi)各組件間输钩、組件與后臺(tái)線程間的通信。比如請(qǐng)求網(wǎng)絡(luò)仲智,等網(wǎng)絡(luò)返回時(shí)通過(guò)Handler或Broadcast通知UI.
作為一個(gè)消息總線买乃,有三個(gè)主要的元素:
- Event:事件
- Subscriber:事件訂閱者,接收特定的事件钓辆;有onEvent剪验,onEventMainThread,onEventBackgroundThread岩馍,onEventAsync這四個(gè)方法碉咆,分別對(duì)應(yīng)不同的處理模式;
- Publisher:事件發(fā)布者蛀恩,用于通知Subscriber有事件發(fā)生
事件處理有4種模式:
- PostThread:事件的處理在和事件的發(fā)送在相同的進(jìn)程疫铜,所以事件處理時(shí)間不應(yīng)太長(zhǎng),不然影響事件的發(fā)送線程双谆,而這個(gè)線程可能是UI線程壳咕。對(duì)應(yīng)的函數(shù)名是onEvent席揽。
- MainThread: 事件的處理會(huì)在UI線程中執(zhí)行。事件處理時(shí)間不能太長(zhǎng)谓厘,這個(gè)不用說(shuō)的幌羞,長(zhǎng)了會(huì)ANR的,對(duì)應(yīng)的函數(shù)名是onEventMainThread竟稳。
- BackgroundThread:事件的處理會(huì)在一個(gè)后臺(tái)線程中執(zhí)行属桦,對(duì)應(yīng)的函數(shù)名是onEventBackgroundThread,雖然名字是BackgroundThread他爸,事件處理是在后臺(tái)線程聂宾,但事件處理時(shí)間還是不應(yīng)該太長(zhǎng),因?yàn)槿绻l(fā)送事件的線程是后臺(tái)線程诊笤,會(huì)直接執(zhí)行事件系谐,如果當(dāng)前線程是UI線程,事件會(huì)被加到一個(gè)隊(duì)列中讨跟,由一個(gè)線程依次處理這些事件纪他,如果某個(gè)事件處理時(shí)間太長(zhǎng),會(huì)阻塞后面的事件的派發(fā)或處理晾匠。
- Async:事件處理會(huì)在單獨(dú)的線程中執(zhí)行茶袒,主要用于在后臺(tái)線程中執(zhí)行耗時(shí)操作,每個(gè)事件會(huì)開(kāi)啟一個(gè)線程(有線程池)混聊,但最好限制線程的數(shù)目弹谁。適合于長(zhǎng)時(shí)間的后臺(tái)網(wǎng)絡(luò)請(qǐng)求。
使用步驟:
- 定義事件類型:
public class MyEvent {}
- 定義事件處理方法:
通過(guò)注解的方式實(shí)現(xiàn)處理方法定義,具體處理方法名不限定
@Subscribe(threadMode = ThreadMode.MAIN)
public void onEventMainThread
- 注冊(cè)訂閱者:
一般是在event的處理類中會(huì)進(jìn)行eventbus的注冊(cè)句喜,例如Activity或者Fragment中進(jìn)行注冊(cè)
EventBus.getDefault().register(this)
- 發(fā)送事件:
在VM或者P層處理完耗時(shí)操作后post event预愤,讓event訂閱者去處理
EventBus.getDefault().post(new MyEvent())
- 取消注冊(cè):
在activity的onDestroy()方法中要取消注冊(cè),如果一直不釋放會(huì)導(dǎo)致Bus中的Event隊(duì)列內(nèi)存泄露
eventBus.unregister(this);
Handler與EventBus區(qū)別
EventBus是一款針對(duì)Android優(yōu)化的發(fā)布/訂閱事件總線咳胃。主要功能是替代Intent,Handler,BroadCast在Fragment植康,Activity,Service展懈,線程之間傳遞消息.優(yōu)點(diǎn)是開(kāi)銷小销睁,代碼更優(yōu)雅。以及將發(fā)送者和接收者解耦存崖。
通過(guò)demo分別寫(xiě)了Handler和Eventbus對(duì)線程消息傳遞的處理冻记,大家可以簡(jiǎn)單對(duì)比下兩者的區(qū)別,可以發(fā)現(xiàn)EventBus進(jìn)行了完全的解耦来惧。
本文中和demo中使用的是greenrobot的EventBus冗栗,github地址如下:
https://github.com/greenrobot/EventBus