BroadcastReceive基础知识总结
1、BroadcastReceive简介
- BroadcastReceive也就是“广播接收者”的意思,顾名思义,就是用来接收来自系统和应用中的广播
- 在Android系统中,广播体现在方方面面,例如当开机完成后系统会产生一条广播,接收到这条广播就能实现开机启动服务的功能,当网络状态改变时,系统会产生一条广播,接收到这条广播,就能及时的做出提示和保存数据等操作,当电池的电量改变的时候,系统会产生一条广播,接收到这条广播就能在电量低的时候告知用户,及时保存进度。等等
- Android中的广播机制设计的非常出色,很多事情原本需要开发者亲自操作的额,现在只需要广播告知自己就可以了,大大的减少了开发的工作量和开发周期,而作为应用开发者,就需要熟练掌握Android系统提供的一个开发利器,那就是BroadcastReceive。
2、广播接收器的类型
- (1)Normal broadcasts:默认广播
- 发送一条默认的广播 Context.sendBroadcast()方法,普通广播对于多个接收者来说是异步的,通常每个接受者都无需等待即可以接收到广播,接受者相互之间不会有影响,对于这种广播,接受者无法终止广播,即无法阻止其他接受者的接收动作
- MainActivity.java
sendNormal = (Button) findViewById(R.id.sendNormal);
sendNormal.setOnClickListener(new OnClickListener());
/**
* 自定义的内部click
*/
class OnClickListener implements View.OnClickListener {
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.sendNormal:
Intent intent = new Intent("com.wuyinlei.action.BROADCAST");
intent.putExtra("info","我们很有缘,谢谢你收到我");
MainActivity.this.sendBroadcast(intent);
break;
}
}
}
MyReceiver.java
/**
* 自定义的广播接收器
*/
public class MyReceiver extends BroadcastReceiver {
public MyReceiver() {
}
/**
* 接受的方法
* @param context
* @param intent
*/
@Override
public void onReceive(Context context, Intent intent) {
String info = intent.getStringExtra("info");
Toast.makeText(context, info, Toast.LENGTH_SHORT).show();
}
}
mainfest.xml
<receiver
android:name=".MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.wuyinlei.action.BROADCAST"></action>
</intent-filter>
</receiver>
- (2)Ordered broadcasts:有序广播
- 发送一个有序广播使用Context.sendOrderedBroadcast()方法,有序广播比较特殊,它每次只发死你个到优先级较高的接受者那里,然后由优先级高的接受者在传播到优先级别低的接受者那里,优先级高的接受者又能力终止这个广播
- (3)Sticky Broadcast:粘性广播
- 当处理完之后的Intent,依然存在,知道你把它去掉
3、广播接收器的创建步骤
- 1、构建Intent,使用sendBroadcast方法发送广播
- 2、定义一个广播接收器,该广播接收器集成BroadcastReceiver,并且覆盖onReceive()方法来响应事件
- 3、注册广播接收器,我们可以在代码中注册,也可以在AndroidManifest.xml配置文件中注册
4、注册广播接收器的两种方式
- 静态注册
- 静态注册是在AndroidManifest.xml文件中配置
<receiver
android:name=".MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.yinlei.action.MY_BROADCAST"></action>
</intent-filter>
</receiver>
这个时候,我们需要在Activity或者Service中这样
Intent intent = new Intent("com.yinlei.action.MY_BROADCAST");
intent.putExtra("info","我是自定义广播");
sendBroadcast(intent);
- 动态注册
- 动态注册需要在代码中动态的指定广播地址并注册,通常我们是在Activity或者Service注册一个广播
MyReceiver receiver = new MyReceiver();
IntentFilter filter = new IntentFilter();
filter.addAction("android.intent.action.MY_BROADCAST");
registerReceiver(receiver,filter);
解除注册
unregisterReceiver(receiver);
- 注意:
- 在这个方法中发来的广播中,代码注册方式中,接收到的广播的先后和注明优先级最高的他们的先后是随机的,如果没有优先级,代码注册收到为最先。
5、有序广播
- 发送广播
sendOrderedBroadcast();
* 在注册广播中的<intent-filter>中使用android:priority属性,这个属性的范围在-1000---1000。
* 数值越大,优先权越高,在广播接收者中使用setResultExtras方法将一个Bundle对象设置为结果集对象,传递到下一个接受者哪里,这样优先级低的接受者可以用getResultExtras获取到最新的经过处理的信息集合
* 使用sendOrderedBroadcast方法有序广播的时候,需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为null,则表示接收者要接收此广播,需声明指定权限,这样做是从安全角度考虑的,例如系统的短信就是有序的广播的形式,一个应用可能是具有拦截垃圾短信的功能,当短信到来的时候它可以先接收到短信,必要时候终止广播的传递,这样的软件就必须声明接收短信的权限。
- 终止广播传递
abortBroadcast()
* 同级别接收是先后是随机的,在到级别低的收到广播,如果先接收到的把广播拦截了,同级别以外的接收者是无法收到该广播的
* 在这个方法发来的广播(代码注册方式中),收到广播先后次序为:注明优先级的、代码优先级的、没有优先级的,如果都没有优先级,代码注册接收到优先
- 创建的两个有序广播
case R.id.sendOrder:
Intent intentOrder = new Intent("com.wuyinlei.action.MY_BROADCAST");
//第一个参数是intent,第二个是接收的权限
MainActivity.this.sendOrderedBroadcast(intentOrder,null);
break;
mainfest.xml配置
<receiver
android:name=".ThreeReceiver"
android:enabled="true"
android:exported="true"
>
<intent-filter android:priority="100">
<action android:name="com.wuyinlei.action.MY_BROADCAST"></action>
</intent-filter>
</receiver>
<receiver
android:name=".FourReceiver"
android:enabled="true"
android:exported="true">
<intent-filter android:priority="200">
<action android:name="com.wuyinlei.action.MY_BROADCAST"></action>
</intent-filter>
</receiver>
6、粘性广播
- 发送粘性广播使用:sendStickBroadcast(intent)
- 发送这个广播需要权限
<uses-permission android:name="android.permission.BROADCAST_STICKY"/>
- 去掉这个removeStickyBroadcast(intent)
- sendStickyOrderedBroadcast():这个方法具有有序广播的特性也有粘性广播的特性;
- 在这个方法发来的广播中,代码注册方式中,收到广播先后次序为:注明优先级的、代码注册的、没有优先级的;如果都没有优先级,代码注册的收到最先。
MainActivity.java
- 在这个方法发来的广播中,代码注册方式中,收到广播先后次序为:注明优先级的、代码注册的、没有优先级的;如果都没有优先级,代码注册的收到最先。
case R.id.sendSticky:
Intent intentSticky = new Intent("com.wuyinlei.action.MY_BROADCAST_STICKY");
MainActivity.this.sendStickyBroadcast(intentSticky);
break;
case R.id.startSticky:
startActivity(new Intent(MainActivity.this, ReceiveActivity.class));
break;
ReceiveActivity.java
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_receive);
mFiveReceiver = new FiveReceiver();
}
@Override
protected void onResume() {
super.onResume();
IntentFilter filter = new IntentFilter();
filter.addAction("com.wuyinlei.action.MY_BROADCAST_STICKY");
registerReceiver(mFiveReceiver,filter);
}
@Override
protected void onPause() {
super.onPause();
unregisterReceiver(mFiveReceiver);
}
7、接收系统广播
- 1、开机启动广播
- 我们经常会有这样的应用场合,比如消息推送服务,需要实现开机启动的功能,要实现这个功能,我们就可以订阅系统“启动完成”这条广播,接收到这条广播,我们就可以启动自己的服务了
<intent-filter>
<!--注册开机广播地址-->
<action android:name="android.intent.action.BOOT_COMPLETED"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
这个开机广播地址,从安全角度考虑,系统要求必须声明接收开机启动广播的权限。
<user-permission android:name="android.peimission.RECEIVE_BOOT_COMPLETE"/>
* 2、网络状态变化
* 比如用户游览游览器信息的时候,网络突然断开,我们要及时的提醒用户网络已经断开,要实现这个功能,我们可以接受网络状态改变这样的一条广播,当由连接状态变为断开状态时候,系统就会收到一条广播,我们接收到之后,在通过网络的状态做出相应的操作
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
访问的网络状态权限:
<user-permission android:name="android.peimission.ACCESS_NETWORK_STATE"/>
ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo info = cm.getActiveNetworkInfo();
String typeName = info.getTypeName();//连接类型
info.getState();//连接状态
* 3、电量变化
* 如果我们在使用阅读软件,可能全屏阅读,这个时候用户就看不到剩余的电量了,我们就可以为他们提供电量的信息,要想做到这一点,我们需要接受一条电量变化的广播,然后获取百分比信息
* int currLevel = intent.getIntExtra(BatteryManager.EXTRA_LEVEL,0) //当前电量
* int total = intent.getIntExtra(BatteryManager.EXTRA_SCALE,1)//总电量
* int percent = currLevel*100/total;
<intent-filter>
<action android:name="android.intent.action.BATTERY_CHANGED"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
立即获取电量,而不是等电量变化的广播,可以使用:
Intent batteryIntent = getApplicationContext().registerReceiver(null,new IntentFilter(Intent.ACTION_BATTERY_CHANGED))