文章目录
发送一条广播,可以被不同的广播接收者所接收,广播接收者收到广播之后,在进行逻辑处理
一、广播的主要类型
- 标准广播
- 有序广播
- 广播发出去之后,同一时刻只会有一个广播接收器能够收到该条广播,这个接收器中的逻辑执行完毕后,广播才会继续传递。
- 广播接收器有先后顺序,优先级高的广播接收器就可以先收到广播消息,并且可以将广播截断
二、接收广播系统
1、如何创建广播接收器
动态注册
- 新建类,继承BroadcastReceiver,并重写父类的onReceive()方法,有广播来的时候,该方法就会执行。
// 继承自BroadcastReceiver,每当网络发生变化时,,onReceiver方法就会得到执行
private class NetworkChangeReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"network change",Toast.LENGTH_LONG).show();
}
}
- 在活动注册广播
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
EdgeToEdge.enable(this);
setContentView(R.layout.activity_main);
IntentFilter intentFilter = new IntentFilter();
// 广播接收器想要接收什么样的广播,就在这里添加相应的action
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
// 注册使得广播接收器可以监听这条广播
networkChangeReceiver = new NetworkChangeReceiver();
registerReceiver(networkChangeReceiver, intentFilter);
}
当网络发生变化是 Toast会提醒。Android系统为了保护用户设备的安全隐私,做了严格的规定:如果程序需要进行一些对用户来说比较敏感的操作,就必须在配置文件中声明权限才可以,否则程序将直接崩溃。比如这里系统的网络状态就是需要声明的,打开AndroidManifest.xml文件,在里面加入以下权限就可以访问系统网络状态了
<manifest
......
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
</manifest>
ConnectivityManager 是一个系统服务类,专门用来管理网络连接的,调用getActiveNetworkInfo()方法可以得到NetworkInfo实例
// 继承自BroadcastReceiver,每当网络发生变化时,,onReceiver方法就会得到执行
private class NetworkChangeReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
ConnectivityManager connectivityManager = (ConnectivityManager)
getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
if (networkInfo != null && networkInfo.isAvailable()){
Toast.makeText(context,"network change",Toast.LENGTH_LONG).show();
}else{
Toast.makeText(context,"network change",Toast.LENGTH_LONG).show();
}
}
}
当Intent在组件间传递时,组件如果想告知Android系统自己能够响应和处理哪些Intent,那么就需要用到IntentFilter对象。
顾名思义,IntentFilter对象负责过滤掉组件无法响应和处理的Intent,只将自己关心的Intent接收进来进行处理。
IntentFilter实行“白名单”管理,即只列出组件乐意接受的Intent,但IntentFilter只会过滤隐式Intent,显式的Intent会直接传送到目标组件。
Android组件可以有一个或多个IntentFilter,每个IntentFilter之间相互独立,只需要其中一个验证通过则可。除了用于过滤广播的IntentFilter可以在代码中创建外,其他的IntentFilter必须在AndroidManifest.xml文件中进行声明。
静态注册(在AndroidMainfest.xml中注册)
在程序未启动的情况下就能接收到广播,这个接收器是系统自带的,不需要再自定义广播接收器,在第三部分中更加详细的介绍了如何使用静态广播。下面是个实现开机启动程序的小栗子
- 新建广播,右击包->New->Other->Broadcast Receiver
- 静态的广播接收器一定要在AndroidManifest.xml文中注册才可以使用,但是使用快捷方式创建的广播接收器,注册已经被自动完成了,所有静态的广播接收器都是在这里进行注册的。
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
</receiver>
三、发送自定义广播
1、发送标准广播(静态注册)
- 定义一个广播接收器继承BroadcastReceiver。
public class MyBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "received in MyBroadcastReceiver",Toast.LENGTH_LONG).show();
}
}
- 在AndroidManifest.xml 中对这个广播接收器进行修改。
<receiver
android:name=".MyBroadcastReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="com.example.broadcasttest.MY_BROADCAST"/>
</intent-filter>
</receiver>
- 在活动中使用intent发送广播
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
EdgeToEdge.enable(this);
setContentView(R.layout.activity_main);
Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("com.example.broadcasttest.MY_BROADCAST");
intent.setComponent(new ComponentName(getPackageName(),
"com.example.broadcasttest.MyBroadcastReceiver"));
sendBroadcast(intent);
}
});
}
注:第2版的《第一行代码》没有
intent.setComponent(new Component(包名,接收广播的类名));
安卓8.0后的版本,自定义静态广播需要加上这句才能执行。
2、发送有序广播
receiver 标签的属性android:priority
给广播接收器设置优先级。如果在onReceive()方法中调用了abortBroadcast方法,表示将这条广播截断,后面的广播接收器将无法再接收到这条广播。
第二个参数是与权限相关的字符串
sendOrderedBroadcast(intent,null);
四、使用本地广播
本地广播机制发出的广播只能够在用用程序 的内部进行传递,并且广播接收器也只能接收来自本应用程序发出的广播。使用LocalBroadManager来对广播进行管理
五、广播实践——实现强制下线功能
遇到的问题:
-
启动即报错:
One of RECEIVER_EXPORTED or RECEIVER_NOT_EXPORTED should be specified when a receiver isn’t being registered exclusively for system broadcasts
问题解决:在注册的时候加入第三个参数
registerReceiver(receiver, intentFilter, Context.RECEIVER_EXPORTED);
- 广播发送出去了,但是广播接收器未接收到这条广播
问题解决:MainActivity没有继承基类BaseActivity。
分析:如下图
遇到问题首先根据报错信息进行大致定位,分析前后逻辑,打断点。如果看着没啥错误,从头开始捋一遍逻辑,结合继承关系和生命周期。