BroadcastReceiver

前言

  • 本篇主要介绍 广播的类型、注册和发送广播、自定义广播、本地广播等

正文

一、广播的类型

  • 标准广播,是一种完全异步执行的广播,在广播发出之后,几乎所有的广播接收器会在同一时刻接收到这条广播,不分先后顺序,优点是效率高,缺点是在广播的过程中无法被截断

  • 有序广播,是一种同步执行的广播,在广播发出之后,同一时刻只有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑侄子那个完毕之后,广播才会继续传递,根据优先级分先后顺序,优先级高的可以截断与其相邻的优先级低的广播

二、创建一个广播接收者

1
2
3
4
5
6
7
8
//只需要写一个类继承自 BroadcastReceiver 即可,收到广播之后会走 onReceive 方法
public class MyBroadcastReceriver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//注意此方法中不能做耗时操作,其作用更多是为了打开某个组件
}
}

三、注册广播

第二点只是创建了一个广播接收器,具体要接收什么样的广播主要是根据注册广播来决定的。注册广播分两种方式,各有优缺点,分别为 动态注册和静态注册

3.1 动态注册,
1
2
3
4
5
6
7
8
9
10
11
12
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
MyBroadcastReceriver myBroadcastReceriver = new MyBroadcastReceriver();
IntentFilter filter = new IntentFilter();
filter.addAction("android.intent.action.NEW_OUTGOING_CALL");
registerReceiver(myBroadcastReceriver,filter);
  • 如上所示,动态注册了一个监听外拨电话的广播,但是要注意,动态注册广播一定要取消注册,一般是在 onDestroy 方法中 调用 unregisterReceiver(myBroadcastReceriver); 来取消注册

  • 动态注册可以自由地控制注册和取消,比较灵活,但是有一个前提是程序运行之后才会开始接收广播,如果说让接收一条手机开机的广播,动态注册就不行了,这得需要静态注册。

3.2 静态注册

就拿注册一个接收开机启动的广播为例

  • 我们用 AndroidStudio 创建一个广播之后,AndroidManifest.xml 中会添加一个节点,如下:
1
2
3
4
5
<receiver
android:name="broadcastreceiver.MyReceiver"
android:enabled="true"
android:exported="true">
</receiver>
  • 我们需要在这个节点下添加一条 action 属性,来表明接收的是什么样的广播,如下:
1
2
3
4
5
6
7
8
9
10
11
<receiver
android:name="broadcastreceiver.MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"></action>
</intent-filter>
</receiver>

系统开机完成之后,会发一条 android.intent.action.BOOT_COMPLETED 广播,所以我们需要添加这个广播

  • 监听手机开机 还需要添加一条权限
1
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
  • 但是静态注册,不能取消

四、发送自定义广播

  • 之前我们接收的广播都是系统定义好的,而实际项目中,我们很多情况都需要自定义广播。

  • 自定义广播也是分两个步骤,注册广播和发送广播,注册广播也分两种,静态注册和动态注册,跟之前说的完全一样,只不过是那个 action 由我们自己定义而已

4.1 发送标准广播

就拿动态注册来说吧

1
2
3
4
5
//注册
MyBroadcastReceriver myBroadcastReceriver = new MyBroadcastReceriver();
IntentFilter filter = new IntentFilter();
filter.addAction("11111111");
registerReceiver(myBroadcastReceriver,filter);
1
2
3
4
//发送
Intent intent = new Intent();
intent.setAction("11111111");
sendBroadcast(intent);

我们创建一个广播之后,在某个逻辑下 添加如上代码,就表明注册了一个 action 为 11111111 的广播,当有人发送这条广播的时候,我们的 MyBroadcastReceriver 就会收到该广播。

4.2 发送有序广播
  • 相对于标准广播来说,只需要在发送的时候呢改变一行代码即可:
1
2
3
4
sendBroadcast(intent);
//改为
sendOrderedBroadcast(intent,null);
//第二个参数表示一个与权限相关的字符串,我们不需要,传入 null 即可
4.3 设置广播优先级,需要添加一条属性,如下所示:
1
2
3
4
5
6
7
8
9
10
11
12
13
<receiver
android:name="broadcastreceiver.MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter
android:priority="1000"
>
<action android:name="android.intent.action.BOOT_COMPLETED"></action>
</intent-filter>
</receiver>

添加了一条 android:priority=”1000” 属性,其值的范围在 -1000 和 1000 之间,数值越大,优先级越高

4.3 在有序广播中截断广播
  • 只需要在优先级高的广播的 onReceive 方法中调用 abortBroadcast(); 就可截断广播

五、使用本地广播

  • 之前我们所说的所有广播都是全局广播,即发出的广播可以被任何应用程序接收到,而我们的广播接收器,也能接收到来自于其他任何应用程序的广播,特别是我们在发送广播的时候携带一些重要数据,这要是被其他人应用程序获取到了,岂不是很不安全,而本地广播就是为了解决这些问题

  • 本地广播主要是使用了一个 LocalBroadcastManager 类对广播进行管理,如下所示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);
//注册广播
MyBroadcastReceriver myBroadcastReceriver = new MyBroadcastReceriver();
IntentFilter filter = new IntentFilter();
filter.addAction("android.intent.action.NEW_OUTGOING_CALL");
localBroadcastManager.registerReceiver(myBroadcastReceriver,filter);
//发送广播
Intent intent = new Intent();
intent.setAction("11111111");
localBroadcastManager.sendBroadcast(intent);
  • 本地广播除了之前的优点之外,还有一条就是比全局广播更加高效