BroadcastRecevice简介
BroadcastRecevice是Android四大组件之一,广播接收器没有用户界面是一个专注于接收广播通知信息,并做出对应处理的组件。广播事件处理属于系统级的事件处理(一般事件处理是属于View级的事件处理),一个应用可以在发生特定事件时发送Broadcast,系统中的任何应用只要注册了对应的Receiver就会接收到此Broadcast,也就是说一个应用如果对某个广播感兴趣就可以注册对应的Receiver来接收广播,广播事件机制是应用程序(进程间)之间通信的一种手段。
广播类型
- 无序广播(Normal Broadcast)
- 有序广播(Ordered Broadcast)
- 系统广播(System Broadcast)
- 应用内广播(Local Broadcast)
相关API
- Context
- sendBroadcast(Intent intent); //发送无序广播
- sendOrderedBroadcast(Intent intent);//发送有序广播
- registerReceiver(receiver, intentFilter );//注册广播
- unregisterReceiver(receiver);//接触广播
- BroadcastRecevice
- onReceive(Context context,Intent intent);//接收到广播的回调
- abortBroadcast();//中断广播的继续传播
- boolean isOrderedBroadcast();//判断是否是有序广播
注册广播接收器
- 方式一:配置文件注册(静态注册)
/**
* 注册无序广播
* BootBroadcastReceiver 是广播接收者
* UNORDERED_RECEIVER 是为BroadcastReceiver指定action,使之用于接收同action的广播
*/
<receiver android:name=".BootBroadcastReceiver">
<intent-filter>
<action android:name="UNORDERED_RECEIVER" />
</intent-filter>
</receiver>
/**
* 注册有序广播
* BootBroadcastReceiver 是广播接收者
* ORDER_RECEIVER是为BroadcastReceiver指定action,使之用于接收同action的广播
* priority是指当前receiver的优先级,值是0-1000,值越大优先级越大,priority属性相同者,动态注册的广播优先
*/
<receiver android:name=".BootBroadcastReceiver">
<intent-filter android:priority="666">
<action android:name="ORDER_RECEIVER" />
</intent-filter>
</receiver>
- 方式二:硬编码注册(动态注册)
IntentFilter filter=new IntentFilter("ORDER_RECEIVER");
//setPriority(666) 可设 可不设
filter.setPriority(666);
registerReceiver(receiver, filter);
区别静态注册和静态注册
dog | 静态注册 | 动态注册 |
---|---|---|
注册方式 | 配置文件 | java代码 |
注册的时间 | 应用安装成功/手机开机完成,注册时不会创建对象接收到广播才创建 | 执行registerReceiver(receiver,intentFilter),注册就创建对象 |
生命结束的时间(解注册) | 应用卸载,对象执行onReceive后就回收了 | 执行unregisterReceiver(),(activity退出必须解注册)解注册后对象被回收 |
应用场景 | 需要监听的时间为应用的整个生命过程中 | 只服务于某个Activity/Service |
发送广播
- 发送无序广播
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
/**
* UNORDERED_RECEIVER 是注册时的action
* 可以传递intent数据
*/
Intent intent=new Intent("UNORDERED_RECEIVER");
Bundle bundle=new Bundle();
bundle.putString("type", "无序广播");
intent.putExtras(bundle);
//无序广播
sendBroadcast(intent, null);
}
});
- 发送有序广播
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
/**
* ORDER_RECEIVER 是注册时的action
* 可以传递intent数据
*/
Intent intent=new Intent("ORDER_RECEIVER");
Bundle bundle=new Bundle();
bundle.putString("type", "有序广播");
intent.putExtras(bundle);
//有序广播
sendOrderedBroadcast(intent, null);
}
});
广播接收器类
- 广播接收器都必须继承BroadcastReceiver类
public class BootBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
/**
* 可以接收传来的数据
*/
Bundle bundle = intent.getExtras();
String string = bundle.getString("type", "default");
Log.e("TAG", ""+string);
/**
* 判断action 是不是当时注册时写的action:name
*/
if (intent.getAction().equals("ORDER_RECEIVER")){
Intent intent1 = new Intent(context, BillWelcome.class);
intent1.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent1);
}
}
}
解除广播
- 手动解除
//解除广播
public void removeBroadcast(View view){
//unRegisterReceiver(receiver) : 解注册广播接收器
if (receiver!=null) {
unregisterReceiver(receiver);
receiver=null;//置空
Toast.makeText(context, "解除广播", 0).show();
}
}
- onDestroy之前注销
//销毁之前一定要解绑和置空 否则会内存泄漏
@Override
protected void onDestroy() {
super.onDestroy();
if (receiver!=null) {
unregisterReceiver(receiver);
receiver=null;
}
}
普通广播和有序广播
1、Normal Broadcast(无序广播)
Normal Broadcast是完全异步的,可以在同一时刻(逻辑上)被所有接收者接收到,消息传递的效率比较高。但缺点是接受者不能将处理结果传递给下一个接收者,并且无法终止Broadcast Intent的广播。
2、Ordered Broadcast(有序广播)
Ordered Broadcast的接收者将按预先声明的优先级依次接受Broadcast。如:A的级别高于B、B的级别高于C,那么Broadcast先传给A,再传给B,最后传给C。优先级别声明在<intent-filter.../>元素的android:priority属性中,数越大优先级别越高,取值范围为0-1000,优先级别也可以调用IntentFilter对象的setPriority()进行设置。OrderedBroadcast接收者可以终止Broadcast Intent的传播,BroadcastIntent的传播一旦终止(abortBroadcast()终止方法),后面的接收者就无法接收到Broadcast。另外,OrderedBroadcast的接收者可以将数据传递给下一个接收者。如:A得到Broadcast后,可以往它的结果对象中存入数据,当Broadcast传给B时,B可以从A的结果对象中得到A存入的数据。
普通广播 | 有序广播 | |
---|---|---|
多人接收时是否有序 | 无序,都会执行 | 有序,根据优先级和注册先后依次执行 |
是否可以中断 | 不可以 | 可以,通过abortBroadcast()中断后,后面的接收器不能接收到此广播了 |
系统广播
Android中内置了多个系统广播:只要涉及到手机的基本操作(如开机、网络状态变化、拍照等等),都会发出相应的广播。每个广播都有特定的Intent - Filter(包括具体的action),列举几个常见的系统广播action如下。
- 手机开机广播 android.intent.action.BOOT_COMPLETED
- 收到短信广播 android.provider.Telephony.SMS_RECEIVED
- 应用被卸载广播 Intent. ACTION_PACKAGE_REMOVED
- 手机锁屏广播 Intent.ACTION_CLOSE_SYSTEM_DIALOGS
- 手机开屏广播 Intent.ACTION_SCREEN_ON
注册一个系统广播-开机广播
/**
* 注册系统广播 开机广播
* BootBroadcastReceiver 是广播接收者
* android.intent.action.BOOT_COMPLETED 是系统开机广播
* 如果想注册其它系统广播 只需要把action:name改成你想要监听的广播就ok了
*/
<receiver android:name=".bill.api.BootBroadcastReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>
应用内广播(Local Broadcast)
-
跟踪原因:
- Android中的广播可以跨进程甚至跨App直接通信,且注册是exported,对于有intent-filter的情况下默认值是true
/**
* exported 系统默认true
*/
<receiver android:name=".BootBroadcastReceiver"
android:exported="false"
>
<intent-filter>
<action android:name="PJS_BROADCAST" />
</intent-filter>
</receiver> -
安全隐患:
- 其他App针对性发出与当前App intent-filter相匹配的广播,由此导致当前App不断接收广播并处理;
- 其他App注册与当前App一致的intent-filter用于接收广播,获取广播具体信息,即会出现安全性 & 效率性的问题。
-
解决方法:
- 对于同一App内部发送和接收广播,将exported属性人为设置成false,使得非本App内部发出的此广播不被接收;
- 在广播发送和接收时,都增加上相应的permission,用于权限验证;
- intent.setPackage(packageName)指定在,这样此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中。
- Android v4兼容包中给出了封装好的LocalBroadcastManager类,用于统一处理App应用内的广播问题,使用方式上与通常的全局广播几乎相同,只是注册/取消注册广播接收器和发送广播时将主调context变成了LocalBroadcastManager的单一实例。
应用内广播代码实例
public void setReceiver(){
//实例化LocalBroadcastManager的实例
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);
/**
* 注册广播
*/
//实例化BroadcastReceiver的实例
BootBroadcastReceiver receiver=new BootBroadcastReceiver();
IntentFilter filter=new IntentFilter("ORDER_RECEIVER");
//使用LocalBroadcastManager进行动态注册
localBroadcastManager.registerReceiver(receiver, filter);
/**
* 发送广播
*/
Intent intent=new Intent("ORDER_RECEIVER");
//使用LocalBroadcastManager发送广播
localBroadcastManager.sendBroadcast(intent);
}
注意
对于不同注册方式的广播接收器回调OnReceive(Context context,Intent intent)中的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;
网友评论