普通广播(Normal broadcasts)
是完全异步的,可以在同一时刻(逻辑上)被所有接收者接收到,消息传播的效率比较高;
缺点是接收者不能将处理结果传递给下一个接收者,并且无法终止广播Intent的传播;
//发送普通广播,所有订阅者都有机会获得并进行处理
Context.sendBroadcast()
有序广播(Oedered broadcasts)
按照接收者的优先级别(声明在intent-filter元素的android:priority
属性中,数越大优先级越高,取值范围-100~100
;也可以调用IntentFilter
对象的setPriority()
进行设置),被接受者依次接收广播。
前面的接收者有权终止广播(BroadcastReceiver.abortBroadcast()
),如果广播被前面的接收者终止,后面的接收者就无法获取到广播;
前面的接收者可以将处理结果通过setResultExtras(Bundle)
方法存放进结果对象,然后传给下一个接收者,通过Bundle bundle =getResultExtras(true))
可以获取上一个接收者存入在结果对象中的数据。
//发生有序广播
Context.sendOrderedBroadcast()
生命周期
一个广播处理完onReceive()那么系统将认定此对象将不再是一个活动的对象,也就会finished掉。
使用步骤
- 自定义一个类继承
BroadcastReceiver
- 重写
onReceive()
- 在
manifest.xml
中注册
BroadcastReciver生命周期很短
如果需要在onReceiver()
完成一些耗时操作,应该考虑在Service
中开启一个新线程处理耗时操作,不应该在BroadcastReciver
中开启一个新的线程,因为BroadcastReceiver
生命周期很短,在执行完onReceive()
后就结束,如果开启一个新的线程,可能出现BroadcastRecevier
退出了线程还在,而如果BroadcastReceiver
所在的进程结束了,该线程就会标记为一个空线程,根据Android
的内存管理策略,在系统内存紧张的时候,会按照优先级,杀死优先级地的线程,而空线程无异是优先级最低的,这样就可能导致BroadcastReceive
r启动的子线程不能执行完成。
示例
class MyBroadcastReceiver:BroadcastReceiver() {
@Override
override fun onReceive(context: Context?, intent: Intent?) {
if (intent!!.action.equals("test")){
Toast.makeText(context,"test reciver",Toast.LENGTH_LONG).show()
}
}
}
//静态注册
<receiver android:name=".MyBroadcastReceiver">
<intent-filter>
<action android:name="android.intent.action.ACTION_POWER_CONNECTED" />
<action android:name="test"/>//自定义一个广播动作
</intent-filter>
</receiver>
//动态注册
registerReceiver(myBroadcastReceiver,IntentFilter("test"))
//发送普通广播
sendBroadcast(Intent("test"))
静态注册和动态注册的区别
动态注册广播不是常驻型广播,广播随Activity的生命周期。在Activity结束前,移除广播接收者。
静态注册是常驻型广播,当应用程序关闭后,如果有信息广播来,程序会被系统调用自动运行。
- 当广播为有序广播时:
- 优先级高的先接收;
- 同优先级的广播接收者,动态先于静态;
- 同优先级的同类广播接收者,静态:先扫描的优先后扫描的;动态:先注册的优先后注册的;
- 当广播为普通广播时:
- 无视优先级,动态广播接收者优先于静态广播接收者;
- 同优先级的同类广播接收者,静态:先扫描的优先后扫描的;动态:先注册的优先后注册的;
小结
- 在Android 中如果要发送一个广播必须使用sendBroadCast 向系统发送对其感兴趣的广播接收器中。
- 使用广播必须要有一个intent 对象必设置其action动作对象
- 使用广播必须在配置文件中显式的指明该广播对象
- 每次接收广播都会重新生成一个接收广播的对象
- 在BroadCastReceiver中尽量不要处理太多逻辑问题,建议复杂的逻辑交给Activity 或者 Service 去处理
- 如果在AndroidManifest.xml中注册,当应用程序关闭的时候,也会接收到广播。在应用程序中注册就不产生这种情况了。