一 BoardcastReceiver 定义
BoardcastReceiver 中文译为广播接收器,贵为 Android 四大组件之一。以观察者模式实现的一个全局的监听器。分为两个角色:广播发送者和广播接受者。
二 BoardcastReceiver 作用
01 用于监听/接收 应用发出的广播消息,并作出反应。
02 分应用场景分
1 不同组件之间的通信(包括同一应用间和不同应用间)
2 与 Android 系统在特定情况下的通信,如:电话呼入时,短信接收时,网络连接时。
3 多线程通信。
三 BoardcastReceiver 实现原理
广播使用了观察者模式,基于消息的发布/订阅事件模型,将发送者和接收者进行解耦。
在这个消息发布/订阅事件模型中有三个角色:
原理描述:
1 广播接收者通过 Binder 机制在 AMS 注册
2 广播发送者通过 Binder 机制向 AMS 发送广播
3 AMS 根据广播发送者要求(IntentFilter 和 Permission),在已注册列表中,寻找合适的广播接收者
4 AMS 将广播发送到合适的广播接收者相应的消息循环队列中
5 广播接收者通过消息循环拿到此广播,并回调 onReceive()
四 BoardcastReceiver 使用
01 自定义广播接收者 BoardcastReceiver
- 继承自BroadcastReceivre基类
- 必须复写抽象方法onReceive()方法
- 广播接收器接收到相应广播后,会自动回调onReceive()方法
- 一般情况下,onReceive方法会涉及与其他组件之间的交互,如发送Notification、启动service等
- 默认情况下,广播接收器运行在UI线程,因此,onReceive方法不能执行耗时操作,否则将导致ANR。
02 注册广播接收器
1 静态注册
- 在AndroidManifest.xml里通过标签声明, 当此App首次启动时,系统会自动实例化mBroadcastReceiver类,并注册到系统中
实例代码:
<receiver
//这个接收器的开关
android:enabled=["true" | "false"]
//此broadcastReceiver能否接收其他App的发出的广播
//默认值是由receiver中有无intent-filter决定的:如果有intent-filter,默认值为true,否则为false
android:exported=["true" | "false"]
android:icon="drawable resource"
android:label="string resource"
//继承BroadcastReceiver子类的类名
android:name=".mBroadcastReceiver"
//具有相应权限的广播发送者发送的广播才能被此BroadcastReceiver所接收;
android:permission="string"
//BroadcastReceiver运行所处的进程
//默认为app的进程,可以指定独立的进程
//注:Android四大基本组件都可以通过此属性指定自己的独立进程
android:process="string" >
//用于指定此广播接收器将接收的广播类型
//本示例中给出的是用于接收网络状态改变时发出的广播
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
</receiver>
2 动态注册
- 在代码中通过调用Context的registerReceiver()方法进行动态注册BroadcastReceiver
实例代码:
@Override
protected void onResume(){
super.onResume();
//实例化BroadcastReceiver子类 & IntentFilter
mBroadcastReceiver mBroadcastReceiver = new mBroadcastReceiver();
IntentFilter intentFilter = new IntentFilter();
//设置接收广播的类型
intentFilter.addAction(android.net.conn.CONNECTIVITY_CHANGE);
//调用Context的registerReceiver()方法进行动态注册
registerReceiver(mBroadcastReceiver, intentFilter);
}
//注册广播后,要在相应位置记得销毁广播
//即在onPause() 中unregisterReceiver(mBroadcastReceiver)
//当此Activity实例化时,会动态将MyBroadcastReceiver注册到系统中
//当此Activity销毁时,动态注册的MyBroadcastReceiver将不再接收到相应的广播。
@Override
protected void onPause() {
super.onPause();
//销毁在onResume()方法中的广播
unregisterReceiver(mBroadcastReceiver);
}
}
注意:
动态广播最好在Activity的onResume()注册、onPause()注销。
* 原因:*
1 对于动态广播,有注册就必然得有注销,否则会导致内存泄露
2 在onResume()注册、onPause()注销是因为onPause()在App死亡前一定会被执行,从而保证广播在App死亡前一定会被注销,从而防止内存泄露。
03 广播发送者向AMS发送广播
1 广播的发送
- 广播是用”意图(Intent)“标识
- 定义广播的本质:定义广播所具备的“意图(Intent)”
- 广播发送:广播发送者将此广播的”意图“通过sendBroadcast()方法发送出去
2 广播的类型
1. 普通广播(Normal Broadcast)
即开发者自身定义intent的广播(最常用)。发送广播使用如下:
Intent intent = new Intent();
//对应BroadcastReceiver中intentFilter的action
intent.setAction(BROADCAST_ACTION);
//发送广播
sendBroadcast(intent);
若被注册了的广播接收者中注册时intentFilter的action与上述匹配,则会接收此广播(即进行回调onReceive())。如下mBroadcastReceiver则会接收上述广播,若发送广播有相应权限,那么广播接收者也需要相应权限
2. 系统广播(System Broadcast)
- Android中内置了多个系统广播:只要涉及到手机的基本操作(如开机、网络状态变化、拍照等等),都会发出相应的广播
- 每个广播都有特定的Intent - Filter(包括具体的action)。
- 当使用系统广播时,只需要在注册广播接收者时定义相关的action即可,并不需要手动发送广播,当系统有相关操作时会自动进行系统广播
3. 有序广播(Ordered Broadcast)
4. App应用内广播(Local Broadcast)
- 背景
- Android中的广播可以跨App直接通信(exported对于有intent-filter情况下默认值为true)
- 冲突
- 其他App针对性发出与当前App intent-filter相匹配的广播,由此导致当前App不断接收广播并处理;
- 其他App注册与当前App一致的intent-filter用于接收广播,获取广播具体信息;
即会出现安全性 & 效率性的问题。
- 解决方案:使用App应用内广播(Local Broadcast)
- App应用内广播可理解为一种局部广播,广播的发送者和接收者都同属于一个App。
- 相比于全局广播(普通广播),App应用内广播优势体现在:安全性高 & 效率高
使用
- 将全局广播设置成局部广播
1 注册广播时将exported属性设置为false,使得非本App内部发出的此广播不被接收;
2 在广播发送和接收时,增设相应权限permission,用于权限验证;
3 发送广播时指定该广播接收器所在的包名,此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中。- 使用封装好的LocalBroadcastManager类,该类是一个单例模式的类:实现原理是 handler 消息机制。
实例代码:
//注册应用内广播接收器 //步骤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, intentFilter); //取消注册应用内广播接收器 localBroadcastManager.unregisterReceiver(mBroadcastReceiver); //发送应用内广播 Intent intent = new Intent(); intent.setAction(BROADCAST_ACTION); localBroadcastManager.sendBroadcast(intent);
5 粘性广播(Sticky Broadcast)
五 BoardcastReceiver 注意
01 动态注册和静态注册的区别
02 onReceiver 方法中返回 context 的区别
- 对于静态注册(全局+应用内广播),回调onReceive(context, intent)中的context返回值是:ReceiverRestrictedContext;
- 对于全局广播的动态注册,回调onReceive(context, intent)中的context返回值是:Activity Context;
- 对于应用内广播的动态注册(LocalBroadcastManager方式),回调onReceive(context, intent)中的context返回值是:Application Context。
- 对于应用内广播的动态注册(非LocalBroadcastManager方式),回调onReceive(context, intent)中的context返回值是:Activity Context;
六 感谢
本文整理的内容大部分来自于:
作者:Carson_Ho
链接:http://www.jianshu.com/p/ca3d87a4cdf3
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。