作为安卓中的四大组件广播的作用也是非常大的,听名字我们或许就知道他的功能啦!接下来就回顾、总结下这个重要的组件。
BroadcastReceiver是一个全局的监听器,它可以监听安卓系统、其他App、自己App发出的广播。并作出相应的处理。是安卓的四大组件之一。
BroadcastReceiver就是广播接收者,接收广播。
(1)同一app内部的同一组件内的消息通信(单个或多个线程之间)
(2)同一app内部的不同组件之间的消息通信(单个进程)
(3)同一app有不同的组件,且运行在不同的进程中(多个进程之间)
(4)不同app之间的组件之间消息通信(多个进程之间)
(5)Android系统在特定情况下与App之间的消息通信
发布者、订阅者,一听我们就想到了“观察者设计模式”,没错安卓中的广播机制就是基于观察者模式的,只不过安卓中通过“消息中心ASM”来处理沟通广播接收者、发送者。为啥要把ASM设计为消息中心呢?
(1)原理草图
1、广播接收者BroadcastReceiver通过Binder机制向AMS(Activity Manager Service)进行注册;
2、广播发送者通过binder机制向AMS发送广播;
3、AMS查找符合相应条件(IntentFilter/Permission等)的BroadcastReceiver,将广播发送到BroadcastReceiver(一般情况下是Activity)相应的消息循环队列中;
4、消息循环执行拿到此广播,回调BroadcastReceiver中的onReceive()方法。
步骤:
1、自定义类继承BroadcastReceiver
2、重写onReceiver方法
3、等待广播发送者发送广播,这样BroadcastReceiver的onReceiver就会回调。
/**
* Created by sunnyDay on 2019/8/13 18:09
*/
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "收到广播", Toast.LENGTH_SHORT).show();
}
}
(1)动态注册
public class MainActivity extends AppCompatActivity {
@Override
protected void onResume() {
super.onResume();
IntentFilter filter = new IntentFilter();
filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
MyBroadcastReceiver mBroadcastReceiver = new MyBroadcastReceiver();
registerReceiver(mBroadcastReceiver, filter);
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
/**
* 点击按钮发送自定义广播
* */
public void send(View view) {
Intent intent = new Intent();
intent.setAction("sunny.day.custom.broadcast");// 发送者定义 过滤条件
sendBroadcast(intent);
}
@Override
protected void onPause() {
super.onPause();
// activity 销毁时解注册
unregisterReceiver(mBroadcastReceiver);
}
}
说明:
注意:
广播解注册的时机:
不在onCreate() & onDestory() 或 onStart() & onStop()注册、注销是因为:
当系统因为内存不足(优先级更高的应用需要内存)要回收Activity占用的资源时,Activity在执行完onPause()方法后就会被销毁,有些生命周期方法onStop(),onDestory()就不会执行。当再回到此Activity时,是从onCreate方法开始执行。
假设我们将广播的注销放在onStop(),onDestory()方法里的话,有可能在Activity被销毁后还未执行onStop(),onDestory()方法,即广播仍还未注销,从而导致内存泄露。但是,onPause()一定会被执行,从而保证了广播在App死亡前一定会被注销,从而防止内存泄露。
(2)静态注册
在安卓的manifest中通过receiver节点下添加 intent-filter,然后在intent-filter中添加过滤条件。
<receiver android:name=".MyBroadcastReceiver"
android:exported="false">
<intent-filter >
// 自定义一个action
<action android:name="sunny.day.custom.broadcast" />
intent-filter>
receiver>
receiver节点的属性:
1、exported :此BroadcastReceiver能否接收其他App的发出的广播,这个属性默认值有点意思,其默认值是由receiver中有无intent-filter决定的,如果有intent-filter,默认值为true,否则为false。(同样的,activity/service中的此属性默认值一样遵循此规则)同时,需要注意的是,这个值的设定是以application或者application user id为界的,而非进程为界(一个应用中可能含有多个进程)。
2、name :此BroadcastReceiver类名。
3、permission:如果设置,具有相应权限的广播发送方发送的广播才能被此broadcastReceiver所接收。
4、process:四大组件都具有这个属性。为组件指定单独的进程。不指定默认为app默认的进程。
/**
* 点击按钮发送自定义广播(隐式发送)
*/
private fun send1() {
val intent = Intent()
intent.action = "sunny.day.custom.broadcast"
sendBroadcast(intent)
}
/**
* 点击按钮发送自定义广播(显式发送)
*/
private fun send2() {
val intent = Intent(this,MyBroadcastReceiver::class.java)
intent.action = "sunny.day.custom.broadcast"
sendBroadcast(intent)
}
两种方法发送广播,需要注意的是隐式方式发送广播时,静态注册的广播接受者是收不到广播的。如上send1发送广播时MyBroadcastReceiver#onReceive是不会回调的。
虽然静态注册的自定义广播,隐式启动时完全失效。但是我们可以显式启动啊,如上使用send2发送的广播MyBroadcastReceiver#onReceive是可以收到回调的。
安卓8.0开始限制了静态广播的注册建议:如果您的应用以 API 级别 26 或更高级别的平台版本为目标,则不能使用清单为隐式广播(没有明确针对您的应用的广播)声明接收器,但一些不受此限制的隐式广播除外。在大多数情况下,您可以使用调度作业来代替。
因此我们可以知道:
(3)动态静态注册的区别
1、静态注册:
优点:属于常驻广播,清单文件注册后不受任何组件的影响。即使我们的app退出关闭后有广播来了我们的app依旧会被系统调用。
缺点:由于需要时刻监听广播所以耗电、占内存。2、动态注册:
特点:灵活,跟随组件的生命周期变化。组件销毁时必须要解注册广播,否则容易造成内存泄漏。
普通广播就是我们简单自定义的广播。通过intent携带actionActivity组件发送一个广播,想要接收这个普通广播只需给相应的广播接收者注册下即可:
// 发送普通广播
public void send(View view) {
Intent intent = new Intent();
intent.setAction("sunny.day.custom.broadcast");
sendBroadcast(intent);
}
// 广播接收者注册广播
IntentFilter filter = new IntentFilter();
filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
MyBroadcastReceiver mBroadcastReceiver = new MyBroadcastReceiver();
registerReceiver(mBroadcastReceiver, filter);
(2)广播的权限添加
有些广播为了安全或许会添加一些自定义权限,如下我们在清单文件中自定义一个权限:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.sunnyday.broadcastreceiverrevision">
<!-- 自定义权限-->
<permission
android:name="this.is.custom.permission"
android:protectionLevel="signature"/>
</manifest>
注册广播添加权限:
IntentFilter filter = new IntentFilter();
filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
mBroadcastReceiver = new MyBroadcastReceiver();
registerReceiver(mBroadcastReceiver, filter,"this.is.custom.permission",null);
这时我们点击按钮会发现没收到广播。。。。。原因是你的应用程序没有这个"this.is.custom.permission"权限。你需要声明下。和平常一样: 清单文件
申请下即可。
安卓系统定义个好多广播:比如网络变化会发出相应的广播、来电收到短信也会收到相应的广播、手机开关机、电量低等等、、、、、、
(1)使用
我们不需要手动发送这些广播,这些广播是系统再特定的时机发送的。我们只需注册下相应的action即可。
(2)常见的系统广播action
前面我们接收系统的广播,接收自己自定义的广播都是无序广播,只要订阅了发送者的广播,所有接收者都可以同时接收到。没有先后顺序。接下来便看下有序广播。
(1)接收顺序
1、发出广播后,接收者按照优先级顺序接收广播。
2、所以有序广播是针对广播接收者而言的。
(2)广播接收者接收广播的规则
1、按照Priority属相的大小顺序(大的优先级大)
2、代码注册优先于静态注册(同优先级下)
ps:Priority为intent filter的属性,可以代码设置也可以清单文件属性添加。
(3)广播特点
1、广播接收者按照顺序接收广播(优先级)
2、优先级高的接收者接收广播后可以拦截广播,让优先级低的接收不到。
3、优先级高的广播接收者还可以对广播的内容进行修改,这样优先级低的收到的就是修改后的广播内容。
(4)发送
sendOrderedBroadcast(intent,null)
粘性广播,在 android 5.0/api 21中deprecated,不再推荐使用,相应的还有粘性有序广播,同样已经deprecated。
(1)为啥使用app内部广播
Android中的广播可以跨App直接通信(exported对于有intent-filter情况下默认值为true)这会导致安全问题:
1、其他app针对我们广播接收器匹配规则发出相应的action的广播,我们的接收器就接收、处理。
2、其他app注册了与app一致的action便可获取广播信息
(2)安全问题的解决
1、注册广播时将exported属性设置为false,使得非本App内部发出的此广播不被接收;
2、在广播发送和接收时,增设相应权限permission,用于权限验证;
3、发送广播时指定该广播接收器所在的包名,此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中。(intent.setPackage(packageName))
安卓针对上面的解决思路进行了封装:LocalBroadcastManager类
(3)使用LocalBroadcastManager发送应用内的广播
LocalBroadcastManager这种发送方式只能代码注册。
public class MainActivity extends AppCompatActivity {
private MyBroadcastReceiver mBroadcastReceiver;
private static LocalBroadcastManager localBroadcastManager; // 获得单例
@Override
protected void onResume() {
super.onResume();
localBroadcastManager = LocalBroadcastManager.getInstance(this);
IntentFilter filter = new IntentFilter();
filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
mBroadcastReceiver = new MyBroadcastReceiver();
localBroadcastManager.registerReceiver(mBroadcastReceiver, filter); //注册广播
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
/**
* 点击按钮发送自定义 app内部广播
*/
public void send(View view) {
Intent intent = new Intent();
intent.setAction("sunny.day.custom.broadcast");
localBroadcastManager.sendBroadcast(intent); //发送应用内广播
}
@Override
protected void onPause() {
super.onPause();
// activity 销毁时解注册
localBroadcastManager.unregisterReceiver(mBroadcastReceiver);
}
}
和传统的自定义广播区别:
1、使用了LocalBroadcastManager 类来管理
2、通过LocalBroadcastManager 类发送广播
3、通过LocalBroadcastManager 注册送广播
4、通过LocalBroadcastManager 解绑送广播
参考:
android:exported 属性详解
官方文档