今天无意中看到了android中的LocalBroadcastManager,就google了一下,从名字看就是一个本地广播管理工具类,必然跟广播脱不开关系了,因此下面简单整理一下它的使用方法。
在说LocalBroadcastManger之前,先说一下广播(BroadcastReceiver)的安全性问题:
BroadcastReceiver设计的初衷是从全局考虑可以方便应用程序和系统、应用程序之间、应用程序内的通信,所以对单个应用程序而言BroadcastReceiver是存在安全性问题的(恶意程序脚本不断的去发送你所接收的广播) ,解决方法如下:
保证发送的广播要发送给指定的对象
当应用程序发送某个广播时系统会将发送的Intent与系统中所有注册的BroadcastReceiver的IntentFilter进行匹配,若匹配成功则执行相应的onReceive函数。可以通过类似sendBroadcast(Intent, String)的接口在发送广播时指定接收者必须具备的permission或通过Intent.setPackage设置广播仅对某个程序有效。
保证我接收到的广播是指定对象发送过来的
当应用程序注册了某个广播时,即便设置了IntentFilter还是会接收到来自其他应用程序的广播进行匹配判断。对于动态注册的广播可以通过类似registerReceiver(BroadcastReceiver, IntentFilter, String, android.os.Handler)的接口指定发送者必须具备的permission,对于静态注册的广播可以通过android:exported="false"属性表示接收者对外部应用程序不可用,即不接受来自外部的广播。
上面两个问题其实都可以通过LocalBroadcastManager来解决,LocalBroadcastManager只会将广播限定在当前应用程序中
android.support.v4.content.LocalBroadcastManager
看上面就知道这个类是v4包中提供的一个工具类,主要用于在同一个应用内的不同组件间发送Broadcast。
LocalBroadcastManager用法
和系统广播使用方式类似:
先通过LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this); 获取实例
然后通过函数 registerReceiver来注册监听器
通过 sendBroadcast 函数来发送广播
LocalBroadcastManager mLocalBroadcastManager; BroadcastReceiver mReceiver; @Overrideprotected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); IntentFilter filter = new IntentFilter(); filter.addAction("test"); mReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { if (intent.getAction().equals("test")) { //Do Something } } }; mLocalBroadcastManager = LocalBroadcastManager.getInstance(this); mLocalBroadcastManager.registerReceiver(mReceiver, filter); } @Overrideprotected void onDestroy() { mLocalBroadcastManager.unregisterReceiver(mReceiver); super.onDestroy(); }
总结一下android中的几种广播:
普通广播
通过Context.sendBroadcast发送的广播即为普通广播,对于普通广播接收者接收到它的顺序是不定的,所以接收者接收到后无法使用其他接收者对它的处理结果也无法停止它。
OrderedBroadcastReceiver有序广播
通过Context.sendOrderedBroadcast发送的广播即为有序广播,与普通广播的不同在于,接收者是有序接收到广播的并且可以对广播进行修改或是取消广播向下传递。系统根据接收者定义的优先级顺序决定哪个接收者先接收到它,接收者处理完后可以将结果传递给优先级低的接收者也可以停止广播使得其他优先级低的接收者无法接收到该广播。优先级通过android:priority属性定义,数值越大优先级别越高,取值范围:-1000到1000,虽然API文档介绍对sendBroadcast发送的广播无效,不过本人测试同样有效,相同优先级的接收者接收到广播的顺序随机。Android系统收到短信、接到电话后发送的广播都是有序广播,所以可以进行短信或电话的拦截,即取消广播。
PS:有序广播可以在onReceive函数中通过BroadcastReceiver的abortBroadcast接口(这个接口对sendBroadcast发送广播无效)取消广播,通过接口sendOrderedBroadcast(Intent, String, BroadcastReceiver, android.os.Handler, int, String, Bundle)发送的广播即便优先级高的广播取消了广播,接口参数中指定的BroadcastReceiver依然可以在其他接收者处理完后接收到广播。通过BroadcastReceiver的getResultExtras接口获得结果的Bundle再通过Bundle的putString和getString方法修改或获取数据。
LocalBroadcastManager本地广播
LocalBroadcastManager除了能解决BroadcastReceiver进程间安全性问题外,相对Context操作的BroadcastReceiver而言还具有更高的运行效率。
Sticky Broadcast粘性广播
如果发送者发送了某个广播,而接收者在这个广播发送后才注册自己的Receiver,这时接收者便无法接收到刚才的广播,为此Android引入了StickyBroadcast,在广播发送结束后会保存刚刚发送的广播(Intent),这样当接收者注册完Receiver后就可以继续使用刚才的广播。如果在接收者注册完成前发送了多条相同Action的粘性广播,注册完成后只会收到一条该Action的广播,并且消息内容是最后一次广播内容。系统网络状态的改变发送的广播就是粘性广播。
粘性广播通过Context的sendStickyBroadcast(Intent)接口发送,需要添加权限<uses-permission android:name="android.permission.BROADCAST_STICKY"/>
也可以通过Context的removeStickyBroadcast(Intent intent)接口移除缓存的粘性广播。
StickyOrderedBroadcast粘性有序广播
这个就是粘性广播和有序广播的结合了,通过Context的sendStickyOrderedBroadcast接口发送。
参考:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2015/0102/2249.html
http://www.cnblogs.com/trinea/archive/2012/11/09/2763182.html