Android源码分析--广播的注册、发送和接收

1,997 阅读4分钟

广播的注册、发送和接收过程

广播的注册、发送和接收都与AMS有着密不可分的关系。

广播的注册

广播的注册可分为静态注册和动态注册两种,静态注册在应用安装时由PackageManagerService来完成注册过程,下面我主要来分析动态广播注册。

ContextImpl请求AMS注册广播

在这里插入图片描述

  • 当我们需要动态注册广播时,需要调用Context的registerReceiver方法,然后在ContextWrapperregisterReceiver中调用ContextImplregisterReceiver方法,最终会调用其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方法,然后在ContextWrappersendBroadcast中调用ContextImplsendBroadcast方法,最终会调用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方法,最终会调用ReceiverDispatcherperformReceive方法。

  • 在ReceiverDispatcher的performReceive方法中,会构建类型为Args类型的对象,最终通过mActivityThread(H),将Args对象的getRunnable方法获取的Runnable发送到线程的消息队列中执行。在Args对象的Runnable方法中会调用BroadcastReceiver类型的receiver对象的onReceive方法,这样注册的广播接收者就收到了广播并得到了intent。


联系方式

在这里插入图片描述

微信公众号

在这里插入图片描述