Broadcast 广播简介
按照广播类型分为两种,一种是全局广播,另外一种是本地广播
全局广播:就是发出的广播可以被其他任意的应用程序接收,或者可以接收来自其他任意应用程序的广播。
本地广播:是只能在应用程序的内部进行传递的广播,广播接收器也只能接收内部的广播,不能接受其他应用程序的广播
按照广播机制也可以分为两种,标准广播和有序广播
无序广播:所有的接收者都会接收事件,不可以被拦截,不可以被修改。
有序广播:按照优先级,一级一级的向下传递,接收者可以修改广播数据,也可以终止广播事件。
使用广播接收器接收广播
1.定义一个广播类
在开始使用广播之前(也就是接收广播),我们需要定义一个类,使其继承BroadcastReceiver,复写其中的onrecevie方法,onreceieve方法中就是我们想要广播接收器收到广播之后需要处理的操作。(理解继承是理解面向对象程序设计的关键。在Java中,通过关键字extends继承一个已有的类,被继承的类称为父类(超类,基类),新的类称为子类(派生类)。在Java中不允许多继承。)
public class myreceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//在这里写上相关的处理代码,一般来说,不要此添加过多的逻辑或者是进行任何的耗时操作
//因为广播接收器中是不允许开启多线程的,过久的操作就会出现报错
//因此广播接收器更多的是扮演一种打开程序其他组件的角色,比如创建一条状态栏通知,或者启动某个服务
}
}
2.对广播进行注册
注册的方式有两种,一种是动态注册(使用java代码),另外一种则是静态注册(在AndroidMainFest文件中定义)
动态注册的步骤:
(1)在相关的activity文件中new一个刚才我们定义的广播类
(2)new一个intentFilter类,调用其的setAction方法,参数中传入相关值的action
(3)调用context.registerReceiver方法进行注册,方法的第一个参数为广播类,第二个则是intentFilter类
private myreceiver recevier;
private IntentFilter intentFilter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
recevier = new myreceiver();
intentFilter = new IntentFilter();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
//当网络发生变化的时候,系统广播会发出值为android.net.conn.CONNECTIVITY_CHANGE这样的一条广播
registerReceiver(recevier,intentFilter);
}
这里我们将广播类与intentFilter类声明为全局变量,便于之后的注册与注销 。
需要注意的是,动态注册的广播接收器一定要注销,在onDestroy方法中调用unregisterReceiver(recevier);
静态注册的步骤:
在AndroidMainFest中的application标签下加上receiver的子标签
与通过name属性指定注册一个广播类,也就是我们刚才定义的那个广播类,还有enabled与exported属性,enabled代表是否启用这个广播接收器,exported属性表示是否允许这个广播接收器接受本程序以外的广播(这两步可以通过Android Studio自动完成),之后在receiver标签下加上intent-filter标签,设置其的action。
<receiver android:name=".myreceiver"
android:exported="true"
android:enabled="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"> //开机完成后系统广播发出的一条值为android.intent.action.BOOT_COMPLETED的广播
</intent-filter>
</receiver>
使用android studio可以快速实现广播接收器的注册。
3.在AndroidMainFest声明相关的权限
注册完之后,我们还需要在AndroidMAinFest中声明相关的权限。
<uses-permission android:name="android.permission.WRITE_CALENDAR"/>
静态注册和动态注册广播接收器的区别见下表:
使用广播发送者发送自定义广播
这里介绍如何发送自定义广播(全局广播):
Intent intent = new Intent();
ntent.setAction("com.example.mymessage");
//Intent intent = new Intent("com,example.mymessage");
//也可以像注释这样写
sendBroadcast(intent);//发送标准广播
sendOrderedBroadcast(intent,null);//发送有序广播,//意思就是发送值为com.example.mymessage的这样一条广播
之后,我们把intentfilter中的action改为上面的com.example.mymessage可以了,但是,测试的时候我们会发现不能体现出有序广播的定义,我们需要给广播接收器设置一下优先度
如果使用的是动态注册,直接调用intentFilter.setPriority();
如果是静态注册,则是设置intent-fliter的中Priority属性:
<intent-filter android:pripority="100"/>
优先度的大小设置范围为-1000~1000
想要截断广播,只需要在onreceive()方法中调用aboryBroadcast()即可使广播不再传递下去。
使用本地广播
之前所说是属于全局广播,存在有数据安全的问题,使用本地广播,只能在本地应用程序中发送与接收广播,可以起到保护数据安全的作用。
回想一下,之前动态注册广播是通过registerReceiver(receiver,intentFilter)这个方法来注册的,其实,本地广播的使用与其十分类似,之前的步骤均是一样的,就是在后面注册调用的方法不同而已,注册时候调用的是LocalBroadcastManager的registerReceiver方法,之前调用的是context的registerReceiver方法,两者参数就是相同的。
private myreceiver recevier;
private IntentFilter intentFilter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
recevier = new myreceiver();
intentFilter = new IntentFilter();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
//当网络发生变化的时候,系统广播会发出值为android.net.conn.CONNECTIVITY_CHANGE这样的一条广播
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);
localBroadcastManager.registerReceiver(receiver, intentFilter);
}
同样的,我们也是需要在onDestory方法中使用localBrocastManager.unregisterReceiver()方法注销广播,注意下,图片中我没有把localBroadcastManager定义为全局变量(定义在方法(函数)中的变量称为局部变量,只能在方法(函数)中能使用,定义在方法(函数)外面的变量称为全局变量,全局变量在该源文件下所有的方法(函数)中都可以调用)。
发送广播也是类似的:
localBroadcastManager LocalBroadcastManager = LocalBroadcastManager.getInstance(this);
Intent intent = new Intent("com.exple");
localBroadcastManager.senBroadcast(intent); //发送标准广播
localBroadcastManager.senBroadcastSync(intent); //发送有序广播
常用的action(Action:定义匹配动作,属性值为一个字符串)