廣播的注冊适肠、發(fā)送和接收過程
廣播的注冊霍衫、發(fā)送和接收都與AMS有著密不可分的關(guān)系。
廣播的注冊
廣播的注冊可分為靜態(tài)注冊和動態(tài)注冊兩種侯养,靜態(tài)注冊在應(yīng)用安裝時由
PackageManagerService
來完成注冊過程敦跌,下面我主要來分析動態(tài)廣播注冊。
ContextImpl請求AMS注冊廣播
當(dāng)我們需要動態(tài)注冊廣播時逛揩,需要調(diào)用Context的
registerReceiver
方法柠傍,然后在ContextWrapper的registerReceiver
中調(diào)用ContextImpl的registerReceiver
方法,最終會調(diào)用其registerReceiverInternal方法辩稽。在ContextImpl的
registerReceiverInternal
方法中惧笛,首先是和服務(wù)綁定類似的,通過LoadedApk
類型的mPackageInfo對象的getReceiverDispatcher
方法來獲取IIntentReceiver
類型的rd對象逞泄,用于廣播的跨進程通信患整。然后調(diào)用IActivityManager的registerReceiver
方法,最終調(diào)用AMS的registerReceiver
方法喷众,并將IIntentReceiver
類型的rd對象傳入各谚。在AMS的registerReceiver方法中,首先是調(diào)用
getRecordForAppLocked
方法獲取調(diào)用注冊廣播的應(yīng)用程序進程信息到千,然后根據(jù)進程信息獲取對應(yīng)在AMS中存儲的所有粘性廣播的intent昌渤,然后和傳入的參數(shù)filter的粘性廣播進行對比,找到所有匹配的intent存入到allSticky列表中憔四,最終加入到廣播隊列中執(zhí)行膀息。除此之外,在AMS的
registerReceiver
中還調(diào)用了HashMap類型加矛,存放了所有應(yīng)用進程的廣播接收者列表mRegisteredReceivers履婉,通過傳入之前的IIntentReceiver
對象獲取到對應(yīng)的廣播接收者列表ReceiverList,并將其傳入創(chuàng)建BroadcastFilter斟览,用以描述注冊的廣播接收者。最后將BroadcastFilter添加到IntentResolver類型的mReceiverResolver中辑奈,這樣當(dāng)AMS接收到廣播時苛茂,就可以從mReceiverResolver中直接找到對應(yīng)的廣播接收者,從而達到注冊廣播的目的鸠窗。
廣播的發(fā)送
廣播可以發(fā)送多種類型妓羊,包括無序廣播(普通廣播)、有序廣播和粘性廣播稍计。
Android廣播的分類:
1躁绸、 普通(無序)廣播:使用sendBroadcast
發(fā)送廣播。這種廣播可以依次傳遞給各個處理器去處理。
2净刮、 有序廣播:使用sendOrderedBroadcast
發(fā)送廣播剥哑。這種廣播在處理器端的處理順序是按照處理器的不同優(yōu)先級來區(qū)分的,高優(yōu)先級的處理器會優(yōu)先截獲這個消息淹父,并且可以將這個消息刪除株婴。
3、 粘性消息:使用sendStickyBroadcast
發(fā)送廣播暑认。粘性消息在發(fā)送后就一直存在于系統(tǒng)的消息容器里面困介,等待對應(yīng)的處理器去處理,如果暫時沒有處理器處理這個消息則一直在消息容器里面處于等待狀態(tài)蘸际,粘性廣播的Receiver如果被銷毀座哩,那么下次重建時會自動接收到消息數(shù)據(jù)。
注意:普通廣播和粘性消息不能被截獲粮彤,而有序廣播是可以被截獲的根穷。
這里我們以最簡單的普通廣播發(fā)送為例進行分析。
ContextImpl請求AMS發(fā)送廣播
當(dāng)我們需要發(fā)送無序廣播時驾诈,需要調(diào)用Context的
sendBroadcast
方法缠诅,然后在ContextWrapper的sendBroadcast
中調(diào)用ContextImpl的sendBroadcast
方法,最終會調(diào)用AMS的broadcastIntent方法乍迄。在AMS的
broadcastIntent
方法中管引,首先對發(fā)送的廣播進行合法性校驗,然后調(diào)用其broadcastIntentLocked方法闯两。在AMS的
broadcastIntentLocked
方法中做了很多事情褥伴,對廣播做了一系列的處理后,最終調(diào)用broadcastQueueForIntent
構(gòu)建了廣播隊列漾狼,然后新建BroadcastRecord對象并將其傳入廣播隊列中重慢,同時執(zhí)行廣播隊列的scheduleBroadcastLocked方法。
廣播的接收
AMS到BroadcastReceiver接收廣播
在BroadcastQueue的
scheduleBroadcastLocked
方法中逊躁,發(fā)送了類型為BROADCAST_INTENT_MSG類型的消息似踱,并在消息處理中最終調(diào)用了其processNextBroadcastLocked方法,并在其中遍歷存儲了無序廣播的列表稽煤,然后調(diào)用deliverToRegisteredReceiverLocked
將這些無序廣播的信息描述發(fā)送給對應(yīng)的廣播接收者核芽。在BroadcastQueue的
deliverToRegisteredReceiverLocked
方法中主要檢查廣播發(fā)送者和廣播接收者的權(quán)限,并最終會調(diào)用其performReceiveLocked方法酵熙,然后在其方法中調(diào)用ApplicationThread的scheduleRegisteredReceiver方法轧简。在ApplicationThread的
scheduleRegisteredReceiver
方法中會調(diào)用IIntentReceiver
類型的對象receiver的performReceive方法,而IIntentReceiver
是Binder通信的客戶端匾二,InnerReceiver在本地的代理哮独,它會調(diào)用InnerReceiver的performReceive
方法拳芙,最終會調(diào)用ReceiverDispatcher
的performReceive方法。在ReceiverDispatcher的
performReceive
方法中皮璧,會構(gòu)建類型為Args類型的對象舟扎,最終通過mActivityThread(H),將Args對象的getRunnable方法獲取的Runnable發(fā)送到線程的消息隊列中執(zhí)行恶导。在Args對象的Runnable方法中會調(diào)用BroadcastReceiver類型的receiver對象的onReceive
方法浆竭,這樣注冊的廣播接收者就收到了廣播并得到了intent。