点此进入:从零快速构建APP系列目录导图
点此进入:UI编程系列目录导图
点此进入:四大组件系列目录导图
点此进入:数据网络和线程系列目录导图
为了方便于进行系统级别的消息通知, Android 引入了一套类似的广播消息机制,而且 Android 中的广播机制显得很灵活。它是 Android 四大组件之一,主要用于接收系统或者 App 发送的广播事件。与广播配套是用的是广播接收器,它是一种用于响应系统范围广播通知的组件。许多广播都是由系统发起的,当然应用也可以发起广播。尽管广播接收器不会显示用户界面,但它们可以创建状态栏通知,在发生广播事件时提醒用户。但广播接收器更常见的用途只是作为通向其他组件的“通道”,设计用于执行极少量的工作。 例如,它可能会基于事件发起一项服务来执行某项工作。本篇就将对这一机制的方方面面进行详细的讲解。
本节例程下载地址:WillFlowBroadcast
一、广播机制简介
1、广播机制概述
为什么说 Android 中的广播机制更加灵活呢?这是因为 Android 中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自于系统的,例如,通知屏幕已关闭、电池电量不足或已拍摄照片的广播。也可能是来自于其他应用程序的,例如,通知其他应用某些数据已下载至设备,并且可供其使用。当然 Android 提供了一套完整的 API,允许应用程序自由地发送和接收广播。
2、广播机制分类
Android 中的广播主要可以分为两种类型:无序广播 和 有序广播。
无序广播也叫标准广播(Normal broadcasts),它是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何先后顺序可言。这种广播的效率会比较高,但同时也意味着它是无法被截断的。
标准广播的工作流程如下图所示:
有序广播(Ordered broadcasts) 则是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。比如有三个广播接收者 A、B、C,优先级是 A > B > C。那这个消息先传给 A,再传给 B,最后传给 C。每个接收者有权终止广播,比如 B 终止广播,C 就无法接收到。所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播消息了。
有序广播的工作流程如下图所示:
有序广播和无序广播的共同点:
内部得的实现机制是相同的:通过 Android 系统的 Binder 机制实现通信。
3、关于 BroadCastReceiver 的生命周期
- 相比于我们之前提到的Activity和Fragment的生命周期,广播接收者的生命周期是非常短暂的,在接收到广播的时候创建,onReceive()方法结束之后销毁;
- 广播接收者中不要做一些耗时的工作,否则会弹出 Application No Response 错误对话框;
- 最好也不要在广播接收者中创建子线程做耗时的工作,因为广播接收者被销毁后进程就成为了空进程,很容易被系统杀掉;
- 耗时的较长的工作最好放在服务中完成;
掌握了这些基本概念后,我们就可以来尝试一下广播的用法了。
二、接收系统广播
Android 内置了很多系统级别的广播,我们可以在应用程序中通过监听这些广播来得到各种系统的状态信息。比如手机开机完成后会发出一条广播,电池的电量发生变化会发出一条广播,时间或时区发生改变也会发出一条广播等等。如果想要接收到这些广播,就需要使用广播接收器,下面我们就来看一下它的具体用法。
1、动态注册监听网络变化
广播接收器可以自由地对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,并在内部处理相应的逻辑。注册广播的方式一般有两种,在代码中注册和在 AndroidManifest.xml 中注册,其中前者也被称为动态注册,后者也被称为静态注册。
那么该如何创建一个广播接收器呢?其实只需要新建一个类,让它继承自 BroadcastReceiver,并重写父类的 onReceive()方法就行了。这样当有广播到来时, onReceive()方法就会得到执行,具体的逻辑就可以在这个方法中处理。
修改 MainActivity 中的代码,如下所示:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
init();
}
private void init() {
mIntentFilter = new IntentFilter();
mIntentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
mNetworkChangeReceiver = new NetworkChangeReceiver();
registerReceiver(mNetworkChangeReceiver, mIntentFilter);
}
/**
* 接收系统广播:动态的监听网络变化
*/
private class NetworkChangeReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
ConnectivityManager connectionManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectionManager.getActiveNetworkInfo();
if (networkInfo != null && networkInfo.isAvailable()) {
Toast.makeText(context, "网络可用!", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(context, "网络不可用!", Toast.LENGTH_SHORT).show();
}
}
}
@Override
protected void onDestroy