1.定义
是全局的监听器,属于四大组件之一,分为:广播发送者,广播接收者。
2.作用
- 用于监听/接收应用发出的广播消息,并作出相应的处理。
- 应用场景:
- a. 与Android系统在特定情况下的通信
如当电话呼入的时候、网络变化的时候
- b.多线程通信
- c.不用组件之间的通信(包括应用之间/不同应用之间)
3.实现原理
-
广播使用了设计模式中的观察者模式:基于消息的发布/订阅事件模型。
因此广播的发送者和接收者解耦,使系统方便集成,更容易扩展。
-
模型中的三个角色
广播工作原理.png
1.消息的订阅者(广播接收者)
2.消息的发布者(广播发布者)
3.消息中心(AMS 即 Activity Manager Service)
-
原理描述:
1.广播接收者通过 Binder 机制在 AMS 注册
2.广播发送者通过 Binder 机制向 AMS 发送消息
3.AMS 根据广播发送者要求,在已注册的列表中,寻找合适的广播接收者
寻找依据:IntentFileter / permission
4.AMS 将广播发送到合适的广播接收者相应的消息循环队列中
5.广播接收者通过 消息循环 拿到此广播,并回调 onReceiver();
特别注意,广播发送者和广播接收者的执行是异步的,发出的广播不会关心有无接收者接收,也不确定接收者到底是何时才能接收到。
4.具体使用
4.1 自定义广播接收器BroadCastReceiver
- a. 继承broadcastReciver基类
- b.必须复写抽象方法onReceiver()方法
1.广播接收者接收到相应的广播后,会自动调用onReceiver方法
2.一般情况下,onReceiver方法会涉及与其他组件之间的交互,如发送notification,启动Service等。
3.默认情况下,广播接收器运行在UI线程,因此onReceiver不能进行耗时操作。
public class MyBroadCastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//接收广播之后自动执行的方法体
}
}
4.2 注册广播接收器
-
a.静态注册
在AndroidManifast里面通过标签声明。
<!-- <receiver
android:enabled="true" 默认true
//该接收器能否接收到其他app发出的广播
//默认值是由receiver中有无intent-filter决定的:如果有intent-filter默认为ture,否则为false
android:exported="true"
android:icon="@drawable/abc_vector_test"
android:label="@string/app_name"
//具有相应权限的广播发送者发送的广播才能被此接收者接收
android:permission="3"
//默认为app的进程,可以指定自己独立的进程,
//四大组件都通过该属性指定自己指定的进程
android:process="1"
//继承BroadCastReceiver的子类类名
android:name=".MyBroadCastReceiver">
//用于指定此广播接收器将接收到的广播类型
<intent-filter>
// 用于接收网络状态改变时发出的广播
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
</intent-filter>
</receiver>
-->
实例代码:
<receiver
android:name=".MyBroadCastReceiver">
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
</intent-filter>
</receiver>
当此app启动的时候,系统会自动实例化MyBroadCastReceiver类,并注册到系统中去。
-
b.动态注册
在代码中通过调用Context的registerReciver()方法进行动态注册。
@Override
protected void onResume() {
super.onResume();
//实例化BroadcastReceiver子类 & IntentFilter
myBroadCastReceiver = new MyBroadCastReceiver();
IntentFilter intentFilter = new IntentFilter();
//设置接收广播的类型
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
//动态注册广播
registerReceiver(myBroadCastReceiver, intentFilter);
}
private MyBroadCastReceiver myBroadCastReceiver;
@Override
protected void onPause() {
super.onPause();
//解除广播
unregisterReceiver(myBroadCastReceiver);
}
动态广播最好在onResum()方法中注册,在onPause()方法中注销
(不知道为啥非得在那两个方法里面进行?)
1.对于动态广播,有注册就一定要有注销,避免内存泄漏
2.重复注册,重复注销也不允许。
- c.两种注册方法的区别
- 静态注册(常驻广播)
- 特点:常驻,不受任何组件的生命周期影响(应用程序关闭后,如果有信息广播来,程序依旧会调用),缺点:耗电,占内存。
- 应用场景:需要时刻监听。
2.动态注册(动态广播)
- 特点:非常驻,灵活,与组件的生命周期相关(组件结束=广播接收,在组件结束前必须移除广播接收器)
- 应用场景:需要特定事件监听广播
5.广播发送者向AMS发送广播
5.1 广播分类
- 普通广播(Normal Broadcast)
- 系统广播(System Broadcast)
- 有序广播(Ordered Broadcast)
- 粘性广播(Sticky Broadcast
- App应用内广播 (Local Broadcast)
a. 普通广播(Normal Broadcast)
即开发者自定定义的广播(最常用)。
Intent intent = new Intent();
//对应BroadcastReciver中的intentFilter中的action
intent.setAction(BROADCAST_ACTION);
//发送广播
sendBroadcast(intent);
- 若broadcastReciver在配置清单中注册的intentFilter中的action与上面匹配,就会接到广播,进行回调(执行onReciver())。
<receiver
//此广播接收者类是mBroadcastReceiver
android:name=".mBroadcastReceiver" >
//用于接收网络状态改变时发出的广播
<intent-filter>
<action android:name="BROADCAST_ACTION" />
</intent-filter>
</receiver>
b. 系统广播(System Broadcast)
- android 内置了多个广播:开机,网络状态变化,拍照等。
- 每个广播都有特定的Intent-Filter(包括具体的Action)。
c. 有序广播(Ordered Broadcast)
- 定义发送出去的广播接收者按照先后的顺序接收
有序是针对接收者来说的
- 广播接收者接收的顺序规则(同时面向静态和动态注册的广播接收者)
1.按照Priority属性值从大到小排序
2.Priority属性值相同,动态注册的广播优先
d. 应用内广播(Local Broadcast)
- 背景:Android的广播可以跨APP直接通信(exported对于intent-filter情况下默认值为true)
-
冲突可能出现的问题
1.其他App发出与当前APP intent-filter相匹配的广播,导致当前App不断接收广播并处理。
2.可能获取其他app广播信息,出现安全性&效率性问题 - 解决方案:使用本地广播
1.App应用内广播可以理解为局部广播,广播的发送者和接收者都属于同一个app。
2.相较于全局广播(普通广播),本地广播,更安全,和更高效。
-
具体使用:
1.注册广播时将exported属性设置为false,使得非本APP内发出的广播不被接收- 在广播发送和接收时,增设相应权限Permission,用于权限验证。
3.发送广播时指定该广播接收器所在的包名,此广播只会发送到此包中的APP内与之相匹配的有效广播接收器中。
通过intent.setPackage(packageName)指定包名。
e. 黏性广播(Sticky Broadcast)
在Android5.0,API21中已经失效。
网友评论