1 介绍

广播,是一个全局的监听器,属于Android四大组件之一

在 Android 系统中,广播是在组件之间传递数据的一种机制,这些组件可以位于不同的进程中,起到进程间通信的作用。
BroadcastReceiver 是对发送出来的广播进行过滤、接收和响应的组件,首先将要发送的消息和用于过滤的信息(action、category)装入一个 Intent 对象,然后通过调用 Context.sendBroadcast() 或Context.sendOrderBoradcast() 方法把 Intent 对象以广播形式发送出去。广播发送出去后,所有已注册的 BroadcastReceiver 会检查注册时的 IntnetFilter 是否与发送的 Intent 相匹配,若匹配则会调用 BroadcastReceiver 的 onReceiver 方法。
BroadcastReceiver 的生命周期很短,一旦 onReceiver 执行完毕,生命周期就结束了。

2 创建

  1. public class MyBroadcastReceiver extends BroadcastReceiver {
  2. @Override
  3. public void onReceive(Context context, Intent intent) {
  4. if (!TextUtils.isEmpty(intent.getStringExtra("name"))){
  5. Toast.makeText(context, intent.getStringExtra("name"), Toast.LENGTH_SHORT).show();
  6. }
  7. }
  8. }

3 注册方式

广播的注册分为静态注册和动态注册。

3.1. 静态注册

静态注册即指在 AndroidManifest 里面注册。应用一旦启动,它就会常驻进程,不受任何组件的生命周期影响,即使应用程序关闭后,仍能接收广播,耗电同时占内存。

  1. <receiver android:name=".broadcast.MyBroadcastReceiver"
  2. android:exported="false">
  3. <intent-filter>
  4. <action android:name="demo"/>
  5. </intent-filter>
  6. </receiver>
  1. mButton.setOnClickListener(new ClickListenerWrapper(new View.OnClickListener() {
  2. @Override
  3. public void onClick(View v) {
  4. Intent intent = new Intent("demo");
  5. intent.putExtra("name", "Omooo");
  6. sendBroadcast(intent);
  7. }
  8. }));
  • 静态注册在 Android 8 上已经完全失效了,所以不建议在使用静态注册。

    3.2. 动态注册

    动态注册即在代码中通过调用 Context.registerReceiver(BroadcastReceiver,IntentFilter) 来注册,它有几个重载方法,但是都会有这两个参数。一般建议在 Activity 的 onResume 注册,在 onStop 中注销,避免内存泄露。同时也不允许重复注册、重复注销。
  1. private MyBroadcastReceiver mBroadcastReceiver;
  2. @Override
  3. protected void onResume() {
  4. super.onResume();
  5. IntentFilter filter = new IntentFilter();
  6. filter.addAction("demo");
  7. mBroadcastReceiver = new MyBroadcastReceiver();
  8. registerReceiver(mBroadcastReceiver, filter);
  9. }
  10. @Override
  11. protected void onStop() {
  12. super.onStop();
  13. unregisterReceiver(mBroadcastReceiver);
  14. }

4 广播分类

广播可以分为五类:普通广播、系统广播、有序广播、粘性广播和应用内广播。

4.1. 普通广播

即开发者自定义的广播,也是最常用的广播,上面的实例就是普通广播。

4.2. 系统广播

Android中内置的多个系统广播:只要设计到手机的基本操作(如开关机、网络状态变化、拍照等待),都会发生相应的广播。

系统操作 action
监听网络变化 android.net.conn.CONNECTIVITY_CHANGE
关闭或打开飞行模式 Intent.ACTION_AIRPLANE_MODE_CHANGED
充电时或电量发生变化 Intent.ACTION_BATTERY_CHANGED
电池电量低 Intent.ACTION_BATTERY_LOW
电池电量充足(即从电量低变化到饱满时会发出广播) Intent.ACTION_BATTERY_OKAY
系统启动完成后(仅广播一次) Intent.ACTION_BOOT_COMPLETED
按下照相时的拍照按键(硬件按键)时 Intent.ACTION_CAMERA_BUTTON
屏幕锁屏 Intent.ACTION_CLOSE_SYSTEM_DIALOGS
设备当前设置被改变时(界面语言、设备方向等) Intent.ACTION_CONFIGURATION_CHANGED
插入耳机时 Intent.ACTION_HEADSET_PLUG
未正确移除SD卡但已取出来时(正确移除方法:设置—SD卡和设备内存—卸载SD卡) Intent.ACTION_MEDIA_BAD_REMOVAL
插入外部储存装置(如SD卡) Intent.ACTION_MEDIA_CHECKING
成功安装APK Intent.ACTION_PACKAGE_ADDED
成功删除APK Intent.ACTION_PACKAGE_REMOVED
重启设备 Intent.ACTION_REBOOT
屏幕被关闭 Intent.ACTION_SCREEN_OFF
屏幕被打开 Intent.ACTION_SCREEN_ON
关闭系统时 Intent.ACTION_SHUTDOWN
重启设备 Intent.ACTION_REBOOT







上面我们自己随便写的 action=”demo”,所以每次发送广播需要自己手动调用 sendBroadcast,这时候可以携带数据过去,而系统广播并不能自己传递数据。

4.3. 有序广播

有序是针对广播接收者来说的,前面的实例,广播发出去之后,所有满足条件的广播接收者都能通知接收广播。有序广播则是按照广播接收者的 priority 属性值从大到小排序依次接收,同 priority 的动态广播优先于静态广播。同时,前一个广播接收器也可以通过调用 abortBroadcast() 方法截断广播,这样优先级低的广播接收者就无法接收到广播了,也可以修改数据传递,这就带来了安全性问题。
前面说过,强烈不建议使用静态注册,而 priority 属性就是在 xml 文件里面配置的。
发送有序广播使用:

  1. sendOrderedBroadcast(Intent intent, String receiverPermission)

4.4. 粘性广播

在 Android 5.0 上已经失效,不建议使用。

4.5. 应用内广播

之前发送和接收到的广播全都是属于系统全局广播,即发出的广播可以被其他应用接收到,而且也可以接收到其他应用发送出的广播,这样可能会有不安全因素。
当不需要通过 sendBroadcast 来完成跨应用的通信,那么建议使用 LocalBroadcastManager,将会是更加高效、安全的方式,并且对系统带来的影响也更小。
BroadcastReceiver 采用的 Binder 方式实现跨进程通信,而 LocalBroadcastManager 使用 Handler 通信机制。

函数 作用
LocalBroadcastManager.getInstance(this).registerReceiver(BroadcastReceiver,IntentFilter) 注册 Receiver
LocalBroadcastManager.getInstance(this).unregisterReceiver(BroadcastReceiver,IntentFilter) 注销 Receiver
LocalBroadcastManager.getInstance(this).sendBroadcast(Intent) 发送异步广播
LocalBroadcastManager.getInstance(this).sendBroadcastSync(Intent) 发送同步广播

应用内广播,只能动态注册。

  1. //注册应用内广播接收器
  2. //步骤1:实例化BroadcastReceiver子类 & IntentFilter mBroadcastReceiver
  3. mBroadcastReceiver = new mBroadcastReceiver();
  4. IntentFilter intentFilter = new IntentFilter();
  5. //步骤2:实例化LocalBroadcastManager的实例
  6. localBroadcastManager = LocalBroadcastManager.getInstance(this);
  7. //步骤3:设置接收广播的类型
  8. intentFilter.addAction(android.net.conn.CONNECTIVITY_CHANGE);
  9. //步骤4:调用LocalBroadcastManager单一实例的registerReceiver()方法 进行动态注册
  10. localBroadcastManager.registerReceiver(mBroadcastReceiver, inten tFilter);
  11. //取消注册应用内广播接收器
  12. localBroadcastManager.unregisterReceiver(mBroadcastReceiver);
  13. //发送应用内广播
  14. Intent intent = new Intent(); intent.setAction(BROADCAST_ACTION);
  15. localBroadcastManager.sendBroadcast(intent);

5 实现原理

5.1. 采用的模型

Android中的广播使用了设计模式中的观察者模式:基于消息的发布/订阅事件模型
因此,Android将广播的发送者和接受者解耦,使得系统方便集成,更易扩展。

5.2. 模型讲解

  • 模型中的三个角色
    1. 消息订阅者(广播接收者)
    2. 消息发布者(广播发布者)
    3. 消息中心(AMS,即Activity Manager Service)
  • 原理图

image.png

  1. 广播接收者通过Binder机制在AMS注册
  2. 广播发送者通过Binder机制向AMS发送广播
  3. AMS根据广播发送者要求,在已注册列表中,寻找合适的广播接收者。
  4. AMS将广播发送到合适的广播接收者相应的消息循环队列中
  5. 广播接收者通过消息循环,拿到此广播,并回调onReceive()
  • 广播发送者和广播接收者的执行是一步的,即广播发送者不会关心有无接收者,也不确定接收者合适才能接收到;

6 使用流程

image.png

6.1. 自定义广播接收者BroadcastReceiver

  1. public class mBroadcastReceiver extends BroadcastReceiver {
  2. // 必须复写抽象方法onReceive()方法 接收到广播后自动调用该方法
  3. /**
  4. * 1. 广播接收器接收到相应广播后,会自动回调onReceive()方法
  5. * 2. 一般情况下,onReceive方法会涉及与其他组件之间的交互,如发送 Notification、启动service等
  6. * 3. 默认情况下,广播接收器运行在UI线程,因此,onReceive方法不能 执行耗时操作,否则将导致ANR。
  7. */
  8. @Override public void onReceive(Context context, Intent intent) {
  9. //写入接收广播后的操作
  10. }
  11. }

6.2 广播接收器注册

6.3 广播发送者向AMS发送广播

7 广播的发送

  • 广播是用“意图(Intent)”标识
  • 定义广播的本质:定义广播所具备的“意图(Intent)”
  • 广播发送:广播发送者将此广播的”意图“通过sendBroadcast()方法发送出去