BroadcastReceiver
为了方便进行系统级别通知,android引入了一套广播消息机制,也就是BroadcastReceiver,在Android开发中BroadcastReceiver的使用是非常广泛的它也是Android的四大组件之一,翻译成汉语就是:”广播接收者”。 Android 系统里定义了各种各样的广播,如电池的使用状态,电话的接收和短信的接收,开机启动都会产生一个广播。当然用户也可以自定义自己的广播,它的使用也非常灵活,于是想就BroadcastReceiver做一番总结,来重新了解下android中广播机制。
广播使用
系统广播
1 . 定义好广播接收器继承BroadcastReceiver,注册广播动作。
2 . 系统在发出Action时,在广播接收器onReceive接收广播的Action并作出处理。不同的广播对于一组不同的Action。
3 . 在manifest.xml文件中注册广播,或者通过registerReceiver方法注册广播,当然unregisterReceiver可以注销广播。自定义广播
1 . 通过 sendBroadCast 或 sendOrderedBroadcast发送广播
2 . 接收广播和上面的一样。
广播类型
android广播机制之所以非常灵活,是因为Android应用程序可以对自己感兴趣的广播进行注册,从而只关注自己感兴趣的内容。而在这些内容中可以是系统广播,也可以是其他应用发的广播信息。根据发送者类型可以将广播机制分为:
1.普通广播(Normal Broadcasts)和有序广播(Ordered Broadcasts)。
2. 系统广播和自定义广播。
首先来了解标准广播和有序广播
- 标准广播
标准广播是一种完全异步的广播机制,发送者在发完广播之后,所有注册了广播的应用都可以同时接受到发送的信息,但是这也就意味着广播无法被拦截。应用发广播时,sendBroadCast表示普通广播,置于用法后面在讲。一般标准的广播流程如下
有序广播 是一种同步广播,广播发出之后,前面接收到的广播可以拦截后面的广播,也就是优先级高的广播可以拦截优先级低的广播。当一个广播的逻辑执行完成之后,后面的逻辑才能继续执行,但是要注意的是,不能在onReceiver方法中,执行耗时操作,否则会出现ANR异常,一般在该onReceiver方法中,执行页面的跳转逻辑,创建状态栏通知,或者开启服务等功能。一般有序广播的流程如下:
系统广播
Android系统内置了很多广播,方便用户注册实时接收系统信息状态变化。下面是一些见到的系统广播(来自http://blog.csdn.net/tmj2014/article/details/7815455):
Intent.ACTION_AIRPLANE_MODE_CHANGED;
//关闭或打开飞行模式时的广播
Intent.ACTION_BATTERY_CHANGED;
//充电状态,或者电池的电量发生变化
//电池的充电状态、电荷级别改变,不能通过组建声明接收这个广播,只有通过Context.registerReceiver()注册
Intent.ACTION_BATTERY_LOW;
//表示电池电量低
Intent.ACTION_BATTERY_OKAY;
//表示电池电量充足,即从电池电量低变化到饱满时会发出广播
Intent.ACTION_BOOT_COMPLETED;
//在系统启动完成后,这个动作被广播一次(只有一次)。
Intent.ACTION_CAMERA_BUTTON;
//按下照相时的拍照按键(硬件按键)时发出的广播
Intent.ACTION_CLOSE_SYSTEM_DIALOGS;
//当屏幕超时进行锁屏时,当用户按下电源按钮,长按或短按(不管有没跳出话框),进行锁屏时,android系统都会广播此Action消息
Intent.ACTION_CONFIGURATION_CHANGED;
//设备当前设置被改变时发出的广播(包括的改变:界面语言,设备方向,等,请参考Configuration.java)
Intent.ACTION_DATE_CHANGED;
//设备日期发生改变时会发出此广播
Intent.ACTION_DEVICE_STORAGE_LOW;
//设备内存不足时发出的广播,此广播只能由系统使用,其它APP不可用?
Intent.ACTION_DEVICE_STORAGE_OK;
//设备内存从不足到充足时发出的广播,此广播只能由系统使用,其它APP不可用?
Intent.ACTION_DOCK_EVENT;
//
//发出此广播的地方frameworks\base\services\java\com\android\server\DockObserver.java
Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE;
////移动APP完成之后,发出的广播(移动是指:APP2SD)
Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE;
//正在移动APP时,发出的广播(移动是指:APP2SD)
Intent.ACTION_GTALK_SERVICE_CONNECTED;
//Gtalk已建立连接时发出的广播
Intent.ACTION_GTALK_SERVICE_DISCONNECTED;
//Gtalk已断开连接时发出的广播
Intent.ACTION_HEADSET_PLUG;
//在耳机口上插入耳机时发出的广播
Intent.ACTION_INPUT_METHOD_CHANGED;
//改变输入法时发出的广播
Intent.ACTION_LOCALE_CHANGED;
//设备当前区域设置已更改时发出的广播
Intent.ACTION_MANAGE_PACKAGE_STORAGE;
//
Intent.ACTION_MEDIA_BAD_REMOVAL;
//未正确移除SD卡(正确移除SD卡的方法:设置--SD卡和设备内存--卸载SD卡),但已把SD卡取出来时发出的广播
//广播:扩展介质(扩展卡)已经从 SD 卡插槽拔出,但是挂载点 (mount point) 还没解除 (unmount)
Intent.ACTION_MEDIA_BUTTON;
//按下"Media Button" 按键时发出的广播,假如有"Media Button" 按键的话(硬件按键)
Intent.ACTION_MEDIA_CHECKING;
//插入外部储存装置,比如SD卡时,系统会检验SD卡,此时发出的广播?
Intent.ACTION_MEDIA_EJECT;
//已拔掉外部大容量储存设备发出的广播(比如SD卡,或移动硬盘),不管有没有正确卸载都会发出此广播?
//广播:用户想要移除扩展介质(拔掉扩展卡)。
Intent.ACTION_MEDIA_MOUNTED;
//插入SD卡并且已正确安装(识别)时发出的广播
//广播:扩展介质被插入,而且已经被挂载。
Intent.ACTION_MEDIA_NOFS;
//
Intent.ACTION_MEDIA_REMOVED;
//外部储存设备已被移除,不管有没正确卸载,都会发出此广播?
// 广播:扩展介质被移除。
Intent.ACTION_MEDIA_SCANNER_FINISHED;
//广播:已经扫描完介质的一个目录
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;
//
Intent.ACTION_MEDIA_SCANNER_STARTED;
//广播:开始扫描介质的一个目录
Intent.ACTION_MEDIA_SHARED;
// 广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。
Intent.ACTION_MEDIA_UNMOUNTABLE;
//
Intent.ACTION_MEDIA_UNMOUNTED
// 广播:扩展介质存在,但是还没有被挂载 (mount)。
Intent.ACTION_NEW_OUTGOING_CALL;
Intent.ACTION_PACKAGE_ADDED;
//成功的安装APK之后
//广播:设备上新安装了一个应用程序包。
//一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_CHANGED;
//一个已存在的应用程序包已经改变,包括包名
Intent.ACTION_PACKAGE_DATA_CLEARED;
//清除一个应用程序的数据时发出的广播(在设置--应用管理--选中某个应用,之后点清除数据时?)
//用户已经清除一个包的数据,包括包名(清除包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_INSTALL;
//触发一个下载并且完成安装时发出的广播,比如在电子市场里下载应用?
//
Intent.ACTION_PACKAGE_REMOVED;
//成功的删除某个APK之后发出的广播
//一个已存在的应用程序包已经从设备上移除,包括包名(正在被安装的包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_REPLACED;
//替换一个现有的安装包时发出的广播(不管现在安装的APP比之前的新还是旧,都会发出此广播?)
Intent.ACTION_PACKAGE_RESTARTED;
//用户重新开始一个包,包的所有进程将被杀死,所有与其联系的运行时间状态应该被移除,包括包名(重新开始包程序不能接收到这个广播)
Intent.ACTION_POWER_CONNECTED;
//插上外部电源时发出的广播
Intent.ACTION_POWER_DISCONNECTED;
//已断开外部电源连接时发出的广播
Intent.ACTION_PROVIDER_CHANGED;
//
Intent.ACTION_REBOOT;
//重启设备时的广播
Intent.ACTION_SCREEN_OFF;
//屏幕被关闭之后的广播
Intent.ACTION_SCREEN_ON;
//屏幕被打开之后的广播
Intent.ACTION_SHUTDOWN;
//关闭系统时发出的广播
Intent.ACTION_TIMEZONE_CHANGED;
//时区发生改变时发出的广播
Intent.ACTION_TIME_CHANGED;
//时间被设置时发出的广播
Intent.ACTION_TIME_TICK;
//广播:当前时间已经变化(正常的时间流逝)。
//当前时间改变,每分钟都发送,不能通过组件声明来接收,只有通过Context.registerReceiver()方法来注册
Intent.ACTION_UID_REMOVED;
//一个用户ID已经从系统中移除发出的广播
//
Intent.ACTION_UMS_CONNECTED;
//设备已进入USB大容量储存状态时发出的广播?
Intent.ACTION_UMS_DISCONNECTED;
//设备已从USB大容量储存状态转为正常状态时发出的广播?
Intent.ACTION_USER_PRESENT;
//
Intent.ACTION_WALLPAPER_CHANGED;
//设备墙纸已改变时发出的广播
接下来看下系统广播具体用法
1.动态注册
public class MainActivity extends AppCompatActivity {
private MyReceiver myReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
IntentFilter intentFilter = new IntentFilter("android.net.conn.CONNECTIVITY_CHANGE");
myReceiver = new MyReceiver();
registerReceiver(myReceiver, intentFilter);
}
public class MyReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
ConnectivityManager manager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo activeNetworkInfo = manager.getActiveNetworkInfo();
if(activeNetworkInfo != null && activeNetworkInfo.isAvailable()){
Toast.makeText(MainActivity.this, "有网", Toast.LENGTH_SHORT).show();
}else {
Toast.makeText(MainActivity.this, "无网", Toast.LENGTH_SHORT).show();
}
}
}
/**
* Activity销毁时要注销广播,否则可能内存溢出
*/
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(myReceiver);
}
}
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
打开Menu->System setting -> Data usage切换Mobile Data来启动和禁止手机网络(不是无线网,测试时一直开着无线网,导致一直没反应,此时无线网要关闭)
效果如图:
2 . 静态注册
广播接收器
public class BootReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context ,"开机完成", Toast.LENGTH_LONG).show();
}
}
清单文件中注册:
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<receiver android:name=".BootReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
结果:
到此系统广播用法差不多就是这样,静态注册与动态注册也已经基本了解。我们一般应用的动态注册可以自由控制注册和注销,但是当应用完全退出时,应用就接受不到广播了,要想接收广播必需打开应用,这也是它的主要缺点。但静态注册即使应用没有开启,也能接收广播,因为广播的注册在清单文件中,android系统只要解析了清单文件,即使不打开应用,也可以接受广播。
自定义广播
有时候我们,并不需要接受系统广播,而是根据需求实现自定义广播,这样可以实现进程通信(IPC)和线程通信。举例: 当你在后台播放一首音乐时,而前台要显示进度条进度,此时就需要实现服务和Activity之间通信,当然通信有很多种方式,但是我们可以向前台定时发广播,从而实时更新进度条信息,这样很方便实现解耦和。
- 普通广播
举例看下用法:
我只举例动态注册,静态和系统广播一样就不多说了
//注册:
//action可以自定义
IntentFilter intentFilter = new
IntentFilter("com.chen.android.MyReceiver");
myReceiver = new MyReceiver();
registerReceiver(myReceiver, intentFilter);
//发广播:
sendBroadcast(new Intent("com.chen.android.MyReceiver"));
//接收
public void onReceive(Context context, Intent intent) {
Toast.makeText(MainActivity.this, "接收到了自定义广播", Toast.LENGTH_SHORT).show();
}
结果如图:
同时你也可以在Intent中携带一些参数信息,比如歌曲进度信息,信息类别和Activity中参数传递一样,就不多说了。
Intent intent = new Intent("com.chen.android.MyReceiver");
intent.putExtra("str", "我在发广播");
sendBroadcast(intent);
- 有序广播
//第一个接收器打印
Toast.makeText(context, "first receiver", Toast.LENGTH_SHORT).show();
//第二个接收器打印
Toast.makeText(context, "sencond receiver", Toast.LENGTH_SHORT).show();
//清单文件注册
<receiver android:name=".MyReceiver">
<intent-filter android:priority="2">
<action android:name="com.chen.android.MyReceiver" />
</intent-filter>
</receiver>
<receiver android:name=".MyReceiver2">
<intent-filter android:priority="1">
<action android:name="com.chen.android.MyReceiver" />
</intent-filter>
</receiver>
//发送有序广播
Intent intent = new Intent("com.chen.android.MyReceiver");
sendOrderedBroadcast(intent, null);
其中 priority代表广播优先级,越高越先被接收,并且拥有拦截低优先级的权限
可以看见运行结果:先打印了“first receiver”, 再打印了“sencond receiver”toast信息,当我在第一个接收器中调用abortBroadcast(),运行结果可知只打印了first receiver的Toast信息,证明了前面那张有序广播的图解。
- 无序广播
就不多做演示了,挺简单的。。。
本地广播
发生和接收的广播全部都是属于系统全局广播,且发出的广播可以被其他任何应用程序接收到,并且可以接收到其他任何应用程序的广播,这样容易引发应用程序的安全性,比如通过广播向应用程序发送垃圾信息,以及发送的一些携带关键性数据的广播有可能被其他应用拦截。这些都是不允许的,在android中引入了一套广播机制,使用这个广播接收器只能接收来自本应用的广播,其他的接收不到,这样就不容易引发系统安全问题,本地广播使用了一个LocalBroadcastmanager对广播进行管理,并提供对广播的注销和注册,看下它的api。
1 .其中getInstance获取LocalBroadcastmanager
2 .registerReceiver注册广播
3 .unregisterReceiver注销广播
4 .sendBroadcast 发送广播
这是常用的api,可以知道 它和动态注册广播的使用方式一样,就不多说了。
最后
本篇总结借鉴了郭大侠的《第一行代码中》的广播章节代码,具体可以查看书籍。