一. 广播
1. 广播是什么?
广播(Broadcast)机制用于进程/线程间通信,因此在我们应用程序内发出的广播,其他的应用程序应该也是可以收到的。
2. 广播用到的场景
同一app内部的同一组件内的消息通信(单个或多个线程之间);
同一app内部的不同组件之间的消息通信(单个进程);
同一app具有多个进程的不同组件之间的消息通信;
不同app之间的组件之间消息通信;
Android系统在特定情况下与App之间的消息通信
3. 广播类型
- 标准广播:
异步执行的广播,广播发出后所有的接收器都会接收到,没有先后顺序,无法被截断。 - 有序广播:
同步执行的广播,同一时刻只能有一个接收器接收到,当该接收器完成之后才能继续传播,有先后顺序,可以被接收器截断。
二. 广播接收器(BroadcastReceiver)
1. 什么是广播接收器?
广播接收器是Andoird中用来接收广播(系统发送的或者其他程序发送的)的组件,是Android的四大组件之一
2. 广播接收器类型
- 静态广播接收器
在AndroidManifest文件中进行注册的广播接收器,常驻型广播,也就是说当应用程序关闭后,如果有信息广播来,程序也会被系统调用自动运行。 - 动态广播接收器
通过Context的registerBroadcastReceiver方法进行注册的广播接收器。动态注册广播不是常驻型广播,也就是说广播跟随程序的生命周期。需要手动注销(调用unregisterBroadcastReceiver方法)。
3. 广播接收器的用法(以下代码均为接收系统广播)
PS:有时候接收系统广播的时候可能会需要权限,此时要记得声明权限
- 定义一个广播接收器
public class BootCompleteReceiver extends BroadcastReceiver {
/*收到广播后会调用onReceive方法*/
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"Boot complete",Toast.LENGTH_LONG).show();
throw new UnsupportedOperationException("Not yet implemented");
}
}
- 声明该接收器能接收的广播(通过IntentFilter实现)
- 静态声明(在manifest的receiver标签中配置intent-filter标签,再添加action标签)
<receiver>
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
- 动态声明(在代码中实例化IntentFilter并为其添加action)
IntentFilter intentFilter = new IntentFilter("android.net.conn.CONNECTIVITY_CHANGE");//实例化IntentFilter并添加action
- 注册广播接收器
- 静态注册
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
- 动态注册
mNetWorkChangeReceiver = new NetWorkChangeReceiver();//实例化BroadcastReceiver
registerReceiver(mNetWorkChangeReceiver,intentFilter);//注册BroadcastReceiver
- 动态注册一定要记得注销
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(mNetWorkChangeReceiver);
}
三. 发送广播(全局广播,所有应用程序都可以接收)
1. 简介:
Android不仅允许系统发送广播,也允许应用程序发送广播。应用程序可以发送标准广播和有序广播。
2. 发送方法:
- 发送普通广播
- 定义广播的内容(通过Intent的action)
Intent intent = new Intent("com.foxnickel.broadcasttest.MY_BROADCAST");//定义内容为com.foxnickel.broadcasttest.MY_BROADCAST的广播
- 调用sendBroadcast()方法进行发送
sendBroadcast(intent);//发送广播
- 发送有序广播
- 定义广播的内容(通过Intent的action)
Intent intent = new Intent("com.foxnickel.broadcasttest.MY_BROADCAST");//定义内容为com.foxnickel.broadcasttest.MY_BROADCAST的广播
- 调用sendOrderedBroadcast()方法进行发送
sendOrderedBroadcast(intent,null);//发送有序广播
3. 注意
- 广播是靠Intent来传播的,因此在发送的过程中可以通过Intent携带一些数据,然后再接收器的onReceice方法里面调用intent的get方法即可得到
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).show();
Toast.makeText(context,intent.getStringExtra("data"),Toast.LENGTH_SHORT).show();//获取intent里的数据
}
- 有序广播的接收可以设置优先级(-1000~1000)
<receiver
android:name=".MyBroadcastReceiver"
android:enabled="true"
android:exported="true">
<intent-filter android:priority="100">//优先级低,后收到
<action android:name="com.foxnickel.broadcasttest.MY_BROADCAST"/>
</intent-filter>
</receiver>
<receiver
android:name=".AnotherReceiver"
android:enabled="true"
android:exported="true">
<intent-filter android:priority="200">//优先级高,先收到
<action android:name="com.foxnickel.broadcasttest.MY_BROADCAST"/>
</intent-filter>
</receiver>
- 优先级高的接收器可以截断广播(abortBroadcast()方法),也可以给后面的广播传递数据(通过Bundle)[参考android-BroadcastReceiver 发送有序广播]
四. 本地广播(在应用内部传递的广播)
1. 本地广播的优势
- 所有数据交换都在本程序内,不必担心数据泄露
- 其他程序无法将广播发送到本程序内部
- 更高效
2. 本地广播要注意的地方
- 只能动态注册,不能静态注册
3. 本地广播的用法
- 实例化LocalBroadcastManager
mLocalBroadcastManager = LocalBroadcastManager.getInstance(this);
- 发送广播(调用LocalBroadcastManager的sendBroadcast方法)
Intent intent = new Intent("com.foxnickel.broadcasttest.LOCAL_BROADCAST");
mLocalBroadcastManager.sendBroadcast(intent);//发送本地广播
- 接收广播
- 新建广播接收器(与普通的广播接收器一样)
class LocalReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"received local broadcast",Toast.LENGTH_SHORT).show();
}
}
- 注册接收器,接收广播(调用LocalBroadcastManager的registerBroadcastReceiver方法)
IntentFilter intentFilter = new IntentFilter("com.foxnickel.broadcasttest.LOCAL_BROADCAST");
LocalReceiver localReceiver = new LocalReceiver();
mLocalBroadcastManager.registerReceiver(localReceiver,intentFilter);
- 记得销毁接收器
@Override
protected void onDestroy() {
super.onDestroy();
mLocalBroadcastManager.unregisterReceiver(localReceiver);
}
五. 注意事项
- 动态注册的广播接收器好像无法接收到其他应用程序发来的广播(不知道是不是自己写错了)
-
静态注册的广播接收器即使app已经退出,主要有相应的广播发出,依然可以接收到,但此种描述自Android 3.1开始有可能不再成立(参考Android总结篇系列:Android广播机制)
- 对于系统广播,由于是系统内部直接发出,无法更改此intent flag值,因此,3.1开始对于静态注册的接收系统广播的BroadcastReceiver,如果App进程已经退出,将不能接收到广播。
- 但是对于自定义的广播,可以通过复写此flag为FLAG_INCLUDE_STOPPED_PACKAGES,使得静态注册的BroadcastReceiver,即使所在App进程已经退出,也能能接收到广播,并会启动应用进程,但此时的BroadcastReceiver是重新新建的。