1、广播的概念---对广播的理解
在我们使用Android手机的时候,我们的手机管家中经常会出现开机自启动某某app,那么对于这个某某APP来说,他是怎么知道系统什么时候开机的呢?还有,系统短信怎么知道收到了短信?以及屏幕点亮与关闭、应用卸载与安装等等。简而言之,一个应用是如何知道系统状态的变化,以及别的应用的变化的呢?这就是Android四大组件之一:BroadcastReceiver(广播接收者)的作用。当系统或者应用发生可能会对别的模块产生影响的事件时,可以发送一条广播告诉别的模块:我变化了,那些可能会受到影响的模块可以接收该广播,并对此作出反应。下面会详细说明BroadcastReceiver的具体用法。
2、Android广播的种类
Android中的广播来源一般有两个,一种是System broadcast,来自于Android系统向外发送;另一种是application broadcast,Android中的各个应用都可以向外发送广播,
Android中的广播又可以分成两种类型:标准广播和有序广播。
2.1 标准广播
这是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播,因此他们之间没有任何先后顺序可言。这也就说明这种广播是没有办法被截断的。
2.2 有序广播
这是一种同步执行的广播,在广播发出去之后,同一时刻只会有一个广播接收器能够收到这条广播,当在这个广播接收器中的逻辑执行完成时,广播才会继续传递,此时广播的传递是有先后顺序的,优先级高的广播接收器能够优先接收到广播,并且接收到广播消息的广播接收器可以截断广播消息,这样下面的广播接收器就无法接收到该广播消息了。
3、广播的注册
3.1 静态注册
1)创建自己的广播,定义一个类,继承于BroadcastReceiver,并重写其onReceive方法,在onReceive方法中可以定义在接收到广播后需要实现的逻辑。代码如下:
public class MyReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
/需要实现的逻辑/
}
}
2)注册,在AndroidManifest.xml文件的application中声明receiver,代码实例如下:
<receiver
android:name=".MyReceiver"
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"></action>
</intent-filter
android:enabled="true"
android:exported="true"></receiver>
其中 android:name=是声明广播接收器,enabled属性是声明是否启动这个广播接收器,exported属性是声明是否允许接收本程序以外的广播。Intent-filter标签中声明的是该广播接收器能够接收的广播。这里声明的是该广播接收器能够接收网络状态变化的广播。
注意:当声明应用被安装或者应用被卸载的广播的时候,我们要声明一下数据类型:
<intent-filter>
<action android:name="android.intent.action.PACKAGE_INSTALL"/>
<action android:name="android.intent.action.PACKAGE_REMOVED"/>
<data android:scheme="package"/>
</intent-filter>
Tips:
1、可以在onReceive方法中通过:
String action=intent.getAction();//获取action的名字
Log.i(TAG,"onReceive:"+action);
来获取当前监听到的action是什么类型的action
2、Android8.0之后很多广播不支持静态注册了,事实上在,google因为电池优化的原因从7.0开始已经对静态注册做出了一些限制(权限),并在8.0使大部分静态注册失效了,极少数的系统广播例如ACTION_BOOT_COMPLETED, ACTION_HEADSET_PLUG, ACTION_CONNECTION_STATE_CHANGED 等广播依旧有效。可以说,未来使用动态注册已经是一种趋势,不过在兼容低版本时,静态注册依旧好用。
3.2 动态注册
1)创建自己的广播,定义一个类,继承于BroadcastReceiver,并重写其onReceive方法,在onReceive方法中可以定义在接收到广播后需要实现的逻辑。代码如下:
public class MyReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
/需要实现的逻辑/
}
}
2)在activity的onCreate方法中新建一个广播接收器对象
mMyReceiver = new MyReceiver();
新建IntentFilter对象来定义要接收哪些广播
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(Intent.ACTION_PACKAGE_REMOVED); intentFilter.addDataScheme("package");
通过registerReceiver(参数1,参数2)方法来注册广播接收器,其中第一个参数是广播接收器对象,第二个参数是IntentFilter对象。
注意:如果使用动态注册的方式注册广播接收器,一定要记得通过unregisterReceiver方法注销广播接收器,因为广播机制其实是观察者模式,会把所有接收到的广播保存在内存中,如果不取消注册,可能会导致内存泄漏。一般取消注销的方法是写在ondestroy生命周期中。
4、onReceive( )方法的执行线程和执行时间限制
onReceive()方法是运行在主线程中的,所以其中不能进行太过耗时的操作如下载等操作,如果需要进行耗时操作,那么需要在onReceiver方法中开启一个新的子线程,在子线程中进行耗时操作。
5、自定义广播
5.1 自定义广播+动态注册
1)首先创建自己的广播,定义一个类,继承于BroadcastReceiver,并重写其onReceive方法
2)新建一个广播:Intent intent = new Intent(MY_ACTION);其中MY_ACTION是一个自定义的常量,用来表示广播的名字。可以通过Intent.putExtra()方法来将要携带的数据放入广播中,最后调用sendBroadcast(intent)方法将广播发送出去。
3)在activity的onCreate方法中新建一个广播接收器对象,设置IntentFilter的对象的Action为自定义的广播名
Demo在附件中的BroadcastDemo3中
5.2 两个应用之间的广播发送和接收
BroadcastDemo3中已经包含向外发送广播的功能,所以现在只需要写一个Demo用来接收广播就可以了。
1)首先创建自己的广播,定义一个类,继承于BroadcastReceiver,并重写其onReceive方法。
2)因为BroadcastDemo3中向外发送广播的时候通过Intent.putExtra()方法将要携带的数据放入了广播中,所以在广播接收器的onReceive方法中可以通过Intent.getStringExtra()方法将广播携带的数据拿到,再通过TextView.setText()方法将得到的字符串类型的数据显示在TextView控件中。
Tips:因为控件所在布局是绑定在Activity中的,所以MyReceiver类中无法直接获取到TextView实例化的对象,这里可以通过给MyReceiver类添加一个含参构造函数,参数是TextView类型的对象,之后在Activity中实例化MyReceiver的时候把TextView对象传进去就可以实现在onReceiver方法中对activty中绑定的TextView控件进行操作。
Demo在附件中的ReceiverDemo中,ReceiverDemo需配合BroadcastDemo3一起使用,BroadcastDemo3发送广播,ReceiverDemo接收广播。