目录
一、Broadcast 的定义
Broadcast 是一种广泛运用的、在应用程序之间传输信息的机制,Android 中的广播与传统意义上的电台广播类似,一个广播可以有任意个接收者,当然也可能不被任何应用程序所接收。广播机制是一个典型的发布-订阅模式,也就是观察者模式。
广播机制最大的特点是发送方不关心接收方是否接受到数据,也不关心接收方是如何处理数据的,通过这样的方式来达到接、收双方的完全解耦。
(一)广播的三要素:
- 广播(Broadcast):用于发送广播
- 广播接收器(BroadcastReceiver):用于接收广播
- 意图内容(Intent):用于保存广播相关信息的媒介
(二)广播的分类:
- 普通广播
- 有序广播
- 本地广播
- Sticky 广播
二、普通广播
普通广播是完全异步的
,通过 Context 的 sendBroadcast() 函数来发送,消息传递的效率比较高,但所有的 receivers(接收器)的执行顺序不确定。缺点是:接收者不能将处理结果传递给下一个接收者,并且无法终止广播 Intent 的传播,直到没有与之匹配的广播接收器为止。
下面我们来写一个普通广播:
step1:定义一个广播接收器
public class MyBroadcastReceiver extends BroadcastReceiver {
private static final String TAG = "MyBroadcastReceiver";
@Override
public void onReceive(Context context, Intent intent) {
if (intent == null) {
return;
}
String name = intent.getStringExtra("name");
Log.d(TAG, name);
}
}
step2:注册广播
注册广播的方式有两种:静态注册 和 动态注册
- 静态注册:即在 AndroidManifest.xml 文件中进行注册
<receiver
android:name=".MyBroadcastReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.cyy.broad" />
</intent-filter>
</receiver>
enabled 设置为 true 意味着能够接受到信息,exported 为 true 意味着能够接收到外部 apk 广播发送的信息。
- 动态注册:即在代码中进行注册
public void registerHelloBroadcast() {
receiver = new MyBroadcastReceiver();
registerReceiver(receiver, new IntentFilter("com.cyy.broad"));
}
如果使用动态注册,那么不要忘记在执行 onDestroy 时注销广播:
@Override
protected void onDestroy() {
super.onDestroy();
if (receiver != null) {
unregisterReceiver(receiver);
}
}
step3:发送广播 sendBroadcast()
如果是静态注册的广播:
Intent intent = new Intent("com.cyy.broad");
ComponentName componentName = new ComponentName("com.chenyouyu.broadcastdemo",
"com.chenyouyu.broadcastdemo.MyBroadcastReceiver");
intent.setComponent(componentName);
intent.putExtra("name", "hello");
sendBroadcast(intent);
如果是动态注册的广播:
Intent intent = new Intent("com.cyy.broad");
intent.putExtra("name", "hello");
sendBroadcast(intent);
当我们调用了 sendBroadcast() 后,会调用 MyBroadcastReceiver 的 onReceive 函数。
三、有序广播
有序广播通过 Context.sendOrderedBroadcast() 来发送,所以的广播接收器按照优先级依次执行,广播接收器的优先级通过 receiver 的 intent-filter 中的 android:priority
属性来设置,数值越大优先级越高(参数为 -1000 ~ 1000)。当广播接收器接收到广播后,可以使用 setResult() 函数来将结果传递给下一个广播接收器,然后通过 getResult() 函数来取得上一个广播接收器返回的结果,并可以使用 abortBroadcast() 函数来让系统丢弃该广播,使该广播不再传递到别的广播接收器。
设置广播的优先级代码示例:
<receiver
android:name=".MyBroadcastReceiver"
android:enabled="true"
android:exported="true">
<intent-filter android:priority="100">
<action android:name="com.cyy.broad" />
</intent-filter>
</receiver>
关于 android:priority 的取值范围,官网给出的是 -1000 ~ 1000 ,但是看到很多人设置成2147483647(Integer.MAX_VALUE)这个值,可能因为 android:priority 的属性值是 integer 类型,系统会拿这个值和其他值做比较,结果怎么都是它最大了。
发送有序广播:
sendOrderedBroadcast(intent,null);
四、本地广播
在 21 版的 Support v4 包中新增本地广播,也就是 LocalBroadcastManager 。之前广播都是全局的,所有应用程序都可以接收到,这样就会带来安全隐患。但是,有的时候我们并不需要把自己的应用内的信息广播给所有应用,而只是进程内使用,现在使用 Support v4 包中的 LocalBroadcastManager 就能够实现限于应用内的广播。
它的用法很简单,只需要把调用 context 的 sendBroadcast、registerReceiver、unregisterReceiver 的地方替换为 LocalBroadcastManager.getInstance(Context context) 中对应的函数即可:
函数 | 作用 |
---|---|
LocalBroadcastManager.getInstance(context).registerReceiver(receiver,intentFilter) | 注册 Receiver |
LocalBroadcastManager.getInstance(context).unregisterReceiver(receiver) | 注销 Receiver |
LocalBroadcastManager.getInstance(context).sendBroadcast(intent) | 发送异步广播 |
LocalBroadcastManager.getInstance(context).sendBroadcastSync(intent) | 发送同步广播 |
五、sticky 广播
sticky 广播通过 Context.sendStickyBroadcast() 函数来发送,用此函数发送的广播会一直滞留,当有匹配此广播的广播接收器被注册后,该广播接收器就会收到此条广播。使用此函数发送广播时,需要获得 BROADCAST_STICKY 权限:
<uses-permission android:name="android.permission.BROADCAST_STICKY"/>
sendStickyBroadcast() 只保留最后一条广播,并且一直保留下去,这样即使已经有广播接收器处理了该广播,当再有匹配的广播接收器被注册时,此广播仍会被接收。如果你只想处理一遍该广播,可以通过 removeStickyBroadcast() 函数实现。