目录
一、BroadcastReceiver作用
二、BroadcastReceiver分类
三、BroadcastReceiver发送
四、BroadcastReceiver接收
五、BroadcastReceiver对进程状态的影响
六、BroadcastReceiver各系统版本的区别
七、最佳实践
本篇是普通知识点总结,源码分析传送门 杨说:BroadcastReceiver源码分析
一、BroadcastReceiver作用
使用观察者模式使android和app间实现广播通知,可跨应用传递
二、BroadcastReceiver分类
2.1 按定义区分
2.1.1 系统广播
系统会在发生各种系统事件时自动发送广播,例如当系统进入和退出飞行模式时。系统广播会被发送给所有同意接收相关事件的应用。
广播消息本身会被封装在一个 Intent 对象中,该对象的操作字符串会标识所发生的事件(例如 android.intent.action.AIRPLANE_MODE)
2.1.2 自定义广播
用户可按自己的业务需求定义自己的广播
2.2 按顺序区分
2.2.1 有序广播
按优先级发送广播,广播可以被拦截
2.2.2 无序广播
发送广播是无序的,广播不可以被拦截
2.2.3 粘性广播
在注册接收器之前发送广播,接收器注册时会受到该广播,已经被弃用了
三、BroadcastReceiver发送
Android 为应用提供三种方式来发送广播:
3.1 发送有序广播
sendOrderedBroadcast(Intent, String) 方法一次向一个接收器发送广播。当接收器逐个顺序执行时,接收器可以向下传递结果,也可以完全中止广播,使其不再传递给其他接收器。接收器的运行顺序可以通过匹配的 intent-filter 的 android:priority 属性来控制;具有相同优先级的接收器将按随机顺序运行。
3.2 发送无序广播
sendBroadcast(Intent) 方法会按随机的顺序向所有接收器发送广播。这称为常规广播。这种方法效率更高,但也意味着接收器无法从其他接收器读取结果,无法传递从广播中收到的数据,也无法中止广播。
3.3 发送本地
LocalBroadcastManager.sendBroadcast 方法会将广播发送给与发送器位于同一应用中的接收器。如果您不需要跨应用发送广播,请使用本地广播。这种实现方法的效率更高(无需进行进程间通信),而且您无需担心其他应用在收发您的广播时带来的任何安全问题。
广播消息封装在 Intent 对象中。Intent 的操作字符串必须提供应用的 Java 软件包名称语法,并唯一标识广播事件。您可以使用 putExtra(String, Bundle) 向 intent 附加其他信息。您也可以对 intent 调用 setPackage(String),将广播限定到同一组织中的一组应用。
3.4 通过权限限制广播
带权限的发送
当您调用 sendBroadcast(Intent, String) 或 sendOrderedBroadcast(Intent, String, BroadcastReceiver, Handler, int, String, Bundle) 时,可以指定权限参数。接收器若要接收此广播,则必须通过其清单中的 标记请求该权限(如果存在危险,则会被授予该权限
您可以指定现有的系统权限(如 SEND_SMS),也可以使用 元素定义自定义权限。有关权限和安全性的一般信息,请参阅系统权限。
注意:自定义权限将在安装应用时注册。定义自定义权限的应用必须在使用自定义权限的应用之前安装。
四、BroadcastReceiver接收
4.1 清单声明接收器(静态注册)
如果您在清单中声明广播接收器,系统会在广播发出后启动您的应用(如果应用尚未运行)
系统软件包管理器会在应用安装时注册接收器。然后,该接收器会成为应用的一个独立入口点,这意味着如果应用当前未运行,系统可以启动应用并发送广播。
系统会创建新的 BroadcastReceiver 组件对象来处理它接收到的每个广播。此对象仅在调用 onReceive(Context, Intent) 期间有效。一旦从此方法返回代码,系统便会认为该组件不再活跃。
注意:如果您的应用以 API 级别 26 或更高级别的平台版本为目标,则不能使用清单为隐式广播(没有明确针对您的应用的广播)声明接收器,但一些不受此限制的隐式广播除外。在大多数情况下,您可以使用调度作业来代替。
不受任何组件生命周期的影响,占用内存,耗电
4.2 上下文注册接收器(动态态注册)
跟随生命周期生效,只要注册上下文有效,上下文注册的接收器就会接收广播。例如,如果您在 Activity 上下文中注册,只要 Activity 没有被销毁,您就会收到广播。如果您在应用上下文中注册,只要应用在运行,您就会收到广播。
注意:要注册本地广播,请调用 LocalBroadcastManager.registerReceiver(BroadcastReceiver, IntentFilter)。
要停止接收广播,请调用 unregisterReceiver(android.content.BroadcastReceiver)。当您不再需要接收器或上下文不再有效时,请务必注销接收器。否则会造成内存泄露
4.3 带权限的接收
如果您在注册广播接收器时指定了权限参数(通过 registerReceiver(BroadcastReceiver, IntentFilter, String, Handler) 或清单中的 标记指定),则广播方必须通过其清单中的 标记请求该权限(如果存在危险,则会被授予该权限),才能向该接收器发送 Intent。
五、BroadcastReceiver对进程状态的影响
BroadcastReceiver 的状态(无论它是否在运行)会影响其所在进程的状态,而其所在进程的状态又会影响它被系统终结的可能性。例如,当进程执行接收器(即当前在运行其 onReceive() 方法中的代码)时,它被认为是前台进程。除非遇到极大的内存压力,否则系统会保持该进程运行。
但是,一旦从 onReceive() 返回代码,BroadcastReceiver 就不再活跃。接收器的宿主进程变得与在其中运行的其他应用组件一样重要。如果该进程仅托管清单声明的接收器(这对于用户从未与之互动或最近没有与之互动的应用很常见),则从 onReceive() 返回时,系统会将其进程视为低优先级进程,并可能会将其终止,以便将资源提供给其他更重要的进程使用。
因此,您不应从广播接收器启动长时间运行的后台线程。onReceive() 完成后,系统可以随时终止进程来回收内存,在此过程中,也会终止进程中运行的派生线程。要避免这种情况,您应该调用 goAsync()(如果您希望在后台线程中多花一点时间来处理广播)或者使用 JobScheduler 从接收器调度 JobService,这样系统就会知道该进程将继续活跃地工作。如需了解详情,请参阅进程和应用生命周期。
以下代码段展示了一个 BroadcastReceiver,它使用 goAsync() 来标记它在 onReceive() 完成后需要更多时间才能完成。如果您希望在 onReceive() 中完成的工作很长,足以导致界面线程丢帧 (>16ms),则这种做法非常有用,这使它尤其适用于后台线程。
六、BroadcastReceiver各系统版本的区别
6.1 Android 9
从 Android 9(API 级别 28)开始,NETWORK_STATE_CHANGED_ACTION 广播不再接收有关用户位置或个人身份数据的信息。 此外,如果您的应用安装在搭载 Android 9 或更高版本的设备上,则通过 WLAN 接收的系统广播不包含 SSID、BSSID、连接信息或扫描结果。要获取这些信息,请调用 getConnectionInfo()。
6.2 Android 8
Android 8.0 从 Android 8.0(API 级别 26)开始,系统对清单声明的接收器施加了额外的限制。 如果您的应用以 Android 8.0 或更高版本为目标平台,那么对于大多数隐式广播(没有明确针对您的应用的广播),您不能使用清单来声明接收器。当用户正在活跃地使用您的应用时,您仍可使用上下文注册的接收器
6.3 Android 7
Android 7.0
Android 7.0(API 级别 24)及更高版本不发送以下系统广播:
ACTION_NEW_PICTURE
ACTION_NEW_VIDEO
此外,以 Android 7.0 及更高版本为目标平台的应用必须使用 registerReceiver(BroadcastReceiver, IntentFilter) 注册 CONNECTIVITY_ACTION 广播。无法在清单中声明接收器。
七、最佳实践
7.1 LocalBroadcastManager
如果您不需要向应用以外的组件发送广播,则可以使用支持库中提供的 LocalBroadcastManager 来收发本地广播。LocalBroadcastManager 效率更高(无需进行进程间通信),并且您无需考虑其他应用在收发您的广播时带来的任何安全问题。本地广播可在您的应用中作为通用的发布/订阅事件总线,而不会产生任何系统级广播开销
7.2 使用动态注册
如果有许多应用在其清单中注册接收相同的广播,可能会导致系统启动大量应用,从而对设备性能和用户体验造成严重影响。为避免发生这种情况,请优先使用上下文注册而不是清单声明。有时,Android 系统本身会强制使用上下文注册的接收器
7.3 请勿使用隐式 intent 广播敏感信息(发送时限定)
任何注册接收广播的应用都可以读取这些信息。您可以通过以下三种方式控制哪些应用可以接收您的广播:
7.3.1 限制包名
在 Android 4.0 及更高版本中,您可以在发送广播时使用 setPackage(String) 指定软件包。系统会将广播限定到与该软件包匹配的一组应用。
7.3.2 限制权限
您可以在发送广播时指定权限
7.3.3 发送本地广播
您可以使用 LocalBroadcastManager 发送本地广播。
7.4 当您注册接收器时,屏蔽恶意广播(接收时限定)
7.4.1 限定权限
您可以在注册接收器时指定权限
7.4.2 清单接收器限定
对于清单声明的接收器,您可以在清单中将 android:exported 属性设置为“false”。这样一来,接收器就不会接收来自应用外部的广播。
7.4.3 接收本地广播
您可以使用 LocalBroadcastManager 限制您的应用只接收本地广播。
7.5 防止重名
广播操作的命名空间是全局性的。请确保在您自己的命名空间中编写操作名称和其他字符串,否则可能会无意中与其他应用发生冲突。
7.6 防止长时间做耗时操作
由于接收器的 onReceive(Context, Intent) 方法在主线程上运行,因此它会快速执行并返回。如果您需要执行长时间运行的工作,请谨慎生成线程或启动后台服务
7.7 请勿从广播接收器启动 Activity
请勿从广播接收器启动 Activity,否则会影响用户体验,尤其是有多个接收器时。相反,可以考虑显示通知。