1 介绍
广播,是一个全局的监听器,属于
Android
四大组件之一
在 Android 系统中,广播是在组件之间传递数据的一种机制,这些组件可以位于不同的进程中,起到进程间通信的作用。
BroadcastReceiver 是对发送出来的广播进行过滤、接收和响应的组件,首先将要发送的消息和用于过滤的信息(action、category)装入一个 Intent 对象,然后通过调用 Context.sendBroadcast() 或Context.sendOrderBoradcast() 方法把 Intent 对象以广播形式发送出去。广播发送出去后,所有已注册的 BroadcastReceiver 会检查注册时的 IntnetFilter 是否与发送的 Intent 相匹配,若匹配则会调用 BroadcastReceiver 的 onReceiver 方法。
BroadcastReceiver 的生命周期很短,一旦 onReceiver 执行完毕,生命周期就结束了。
2 创建
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if (!TextUtils.isEmpty(intent.getStringExtra("name"))){
Toast.makeText(context, intent.getStringExtra("name"), Toast.LENGTH_SHORT).show();
}
}
}
3 注册方式
3.1. 静态注册
静态注册即指在 AndroidManifest 里面注册。应用一旦启动,它就会常驻进程,不受任何组件的生命周期影响,即使应用程序关闭后,仍能接收广播,耗电同时占内存。
<receiver android:name=".broadcast.MyBroadcastReceiver"
android:exported="false">
<intent-filter>
<action android:name="demo"/>
</intent-filter>
</receiver>
mButton.setOnClickListener(new ClickListenerWrapper(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("demo");
intent.putExtra("name", "Omooo");
sendBroadcast(intent);
}
}));
- 静态注册在 Android 8 上已经完全失效了,所以不建议在使用静态注册。
3.2. 动态注册
动态注册即在代码中通过调用 Context.registerReceiver(BroadcastReceiver,IntentFilter) 来注册,它有几个重载方法,但是都会有这两个参数。一般建议在 Activity 的 onResume 注册,在 onStop 中注销,避免内存泄露。同时也不允许重复注册、重复注销。
private MyBroadcastReceiver mBroadcastReceiver;
@Override
protected void onResume() {
super.onResume();
IntentFilter filter = new IntentFilter();
filter.addAction("demo");
mBroadcastReceiver = new MyBroadcastReceiver();
registerReceiver(mBroadcastReceiver, filter);
}
@Override
protected void onStop() {
super.onStop();
unregisterReceiver(mBroadcastReceiver);
}
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 文件里面配置的。
发送有序广播使用:
sendOrderedBroadcast(Intent intent, String receiverPermission)
4.4. 粘性广播
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:实例化BroadcastReceiver子类 & IntentFilter mBroadcastReceiver
mBroadcastReceiver = new mBroadcastReceiver();
IntentFilter intentFilter = new IntentFilter();
//步骤2:实例化LocalBroadcastManager的实例
localBroadcastManager = LocalBroadcastManager.getInstance(this);
//步骤3:设置接收广播的类型
intentFilter.addAction(android.net.conn.CONNECTIVITY_CHANGE);
//步骤4:调用LocalBroadcastManager单一实例的registerReceiver()方法 进行动态注册
localBroadcastManager.registerReceiver(mBroadcastReceiver, inten tFilter);
//取消注册应用内广播接收器
localBroadcastManager.unregisterReceiver(mBroadcastReceiver);
//发送应用内广播
Intent intent = new Intent(); intent.setAction(BROADCAST_ACTION);
localBroadcastManager.sendBroadcast(intent);
5 实现原理
5.1. 采用的模型
Android中的广播使用了设计模式中的观察者模式:基于消息的发布/订阅事件模型
因此,Android将广播的发送者和接受者解耦,使得系统方便集成,更易扩展。
5.2. 模型讲解
- 模型中的三个角色
- 消息订阅者(广播接收者)
- 消息发布者(广播发布者)
- 消息中心(AMS,即Activity Manager Service)
- 原理图
- 广播接收者通过Binder机制在AMS注册
- 广播发送者通过Binder机制向AMS发送广播
- AMS根据广播发送者要求,在已注册列表中,寻找合适的广播接收者。
- AMS将广播发送到合适的广播接收者相应的消息循环队列中
- 广播接收者通过消息循环,拿到此广播,并回调onReceive()
- 广播发送者和广播接收者的执行是一步的,即广播发送者不会关心有无接收者,也不确定接收者合适才能接收到;
6 使用流程
6.1. 自定义广播接收者BroadcastReceiver
public class mBroadcastReceiver extends BroadcastReceiver {
// 必须复写抽象方法onReceive()方法 接收到广播后自动调用该方法
/**
* 1. 广播接收器接收到相应广播后,会自动回调onReceive()方法
* 2. 一般情况下,onReceive方法会涉及与其他组件之间的交互,如发送 Notification、启动service等
* 3. 默认情况下,广播接收器运行在UI线程,因此,onReceive方法不能 执行耗时操作,否则将导致ANR。
*/
@Override public void onReceive(Context context, Intent intent) {
//写入接收广播后的操作
}
}
6.2 广播接收器注册
6.3 广播发送者向AMS发送广播
7 广播的发送
- 广播是用“意图(Intent)”标识
- 定义广播的本质:定义广播所具备的“意图(Intent)”
- 广播发送:广播发送者将此广播的”意图“通过sendBroadcast()方法发送出去