廣播的注冊、發送和接收過程
廣播的注冊、發送和接收都與AMS有著密不可分的關系。
廣播的注冊
廣播的注冊可分為靜態注冊和動態注冊兩種,靜態注冊在應用安裝時由
PackageManagerService
來完成注冊過程,下面我主要來分析動態廣播注冊。
ContextImpl請求AMS注冊廣播
當我們需要動態注冊廣播時,需要調用Context的
registerReceiver
方法,然后在ContextWrapper的registerReceiver
中調用ContextImpl的registerReceiver
方法,最終會調用其registerReceiverInternal方法。在ContextImpl的
registerReceiverInternal
方法中,首先是和服務綁定類似的,通過LoadedApk
類型的mPackageInfo對象的getReceiverDispatcher
方法來獲取IIntentReceiver
類型的rd對象,用于廣播的跨進程通信。然后調用IActivityManager的registerReceiver
方法,最終調用AMS的registerReceiver
方法,并將IIntentReceiver
類型的rd對象傳入。在AMS的registerReceiver方法中,首先是調用
getRecordForAppLocked
方法獲取調用注冊廣播的應用程序進程信息,然后根據進程信息獲取對應在AMS中存儲的所有粘性廣播的intent,然后和傳入的參數filter的粘性廣播進行對比,找到所有匹配的intent存入到allSticky列表中,最終加入到廣播隊列中執行。除此之外,在AMS的
registerReceiver
中還調用了HashMap類型,存放了所有應用進程的廣播接收者列表mRegisteredReceivers,通過傳入之前的IIntentReceiver
對象獲取到對應的廣播接收者列表ReceiverList,并將其傳入創建BroadcastFilter,用以描述注冊的廣播接收者。最后將BroadcastFilter添加到IntentResolver類型的mReceiverResolver中,這樣當AMS接收到廣播時,就可以從mReceiverResolver中直接找到對應的廣播接收者,從而達到注冊廣播的目的。
廣播的發送
廣播可以發送多種類型,包括無序廣播(普通廣播)、有序廣播和粘性廣播。
Android廣播的分類:
1、 普通(無序)廣播:使用sendBroadcast
發送廣播。這種廣播可以依次傳遞給各個處理器去處理。
2、 有序廣播:使用sendOrderedBroadcast
發送廣播。這種廣播在處理器端的處理順序是按照處理器的不同優先級來區分的,高優先級的處理器會優先截獲這個消息,并且可以將這個消息刪除。
3、 粘性消息:使用sendStickyBroadcast
發送廣播。粘性消息在發送后就一直存在于系統的消息容器里面,等待對應的處理器去處理,如果暫時沒有處理器處理這個消息則一直在消息容器里面處于等待狀態,粘性廣播的Receiver如果被銷毀,那么下次重建時會自動接收到消息數據。
注意:普通廣播和粘性消息不能被截獲,而有序廣播是可以被截獲的。
這里我們以最簡單的普通廣播發送為例進行分析。
ContextImpl請求AMS發送廣播
當我們需要發送無序廣播時,需要調用Context的
sendBroadcast
方法,然后在ContextWrapper的sendBroadcast
中調用ContextImpl的sendBroadcast
方法,最終會調用AMS的broadcastIntent方法。在AMS的
broadcastIntent
方法中,首先對發送的廣播進行合法性校驗,然后調用其broadcastIntentLocked方法。在AMS的
broadcastIntentLocked
方法中做了很多事情,對廣播做了一系列的處理后,最終調用broadcastQueueForIntent
構建了廣播隊列,然后新建BroadcastRecord對象并將其傳入廣播隊列中,同時執行廣播隊列的scheduleBroadcastLocked方法。
廣播的接收
AMS到BroadcastReceiver接收廣播
在BroadcastQueue的
scheduleBroadcastLocked
方法中,發送了類型為BROADCAST_INTENT_MSG類型的消息,并在消息處理中最終調用了其processNextBroadcastLocked方法,并在其中遍歷存儲了無序廣播的列表,然后調用deliverToRegisteredReceiverLocked
將這些無序廣播的信息描述發送給對應的廣播接收者。在BroadcastQueue的
deliverToRegisteredReceiverLocked
方法中主要檢查廣播發送者和廣播接收者的權限,并最終會調用其performReceiveLocked方法,然后在其方法中調用ApplicationThread的scheduleRegisteredReceiver方法。在ApplicationThread的
scheduleRegisteredReceiver
方法中會調用IIntentReceiver
類型的對象receiver的performReceive方法,而IIntentReceiver
是Binder通信的客戶端,InnerReceiver在本地的代理,它會調用InnerReceiver的performReceive
方法,最終會調用ReceiverDispatcher
的performReceive方法。在ReceiverDispatcher的
performReceive
方法中,會構建類型為Args類型的對象,最終通過mActivityThread(H),將Args對象的getRunnable方法獲取的Runnable發送到線程的消息隊列中執行。在Args對象的Runnable方法中會調用BroadcastReceiver類型的receiver對象的onReceive
方法,這樣注冊的廣播接收者就收到了廣播并得到了intent。