一、广播概述
Android 应用可以发送或接收来自 Android 系统和其他 Android 应用的广播消息,类似于 发布 - 订阅 设计模式。这些广播是在感兴趣的事件发生时发送的。例如,Android 系统在发生各种系统事件时发送广播(例如,系统启动或设备开始充电时)。应用程序还可以发送自定义广播,例如,向其他应用程序通知他们可能感兴趣的内容(例如,某些新数据已被下载)。
应用可以注册以接收特定的广播。发送广播时,系统会自动将广播路由到已预订接收该特定广播类型的应用。
一般来说,广播可以用来在应用程序之间传递消息。但是,不要轻易在后台响应广播并运行长时间作业,这可能会导致系统性能降低。
二、关于系统广播
系统在发生各种系统事件时自动发送广播,例如:系统切入和切出飞行模式。系统广播被发送到所有订阅接收事件的应用程序。
广播消息本身包装在一个Intent 对象中,该对象的 action 字符串标识发生的事件(例如 android.intent.action.AIRPLANE_MODE
)。Intent 还可以包含额外信息。例如,airplane mode intent
包含一个 boolean
值,表示飞行模式是否打开。
三、系统广播的变化
Android 7.0 及更高版本不再发送以下系统广播。此优化会影响所有应用,而不仅仅是针对 Android 7.0 的应用。
ACTION_NEW_PICTURE
ACTION_NEW_VIDEO
定位在 Android 7.0(API 级别 24)及更高版本的应用必须通过以下方式注册以下广播 registerReceiver(BroadcastReceiver, IntentFilter)
。在 manifest 中声明 receiver 不再起作用。
CONNECTIVITY_ACTION
从Android 8.0(API 级别 26)开始,系统对 manifest 声明的 receiver 施加额外的限制。如果你的应用定位到 API 级别 26 或更高级别,则无法使用 manifest 为大多数隐式广播声明 receiver。
四、接收广播
应用程序可以通过两种方式接收广播:
通过清单声明接收器;
通过 context 注册接收器。
1. Manifest-declared receivers 清单声明接收器
如果你在清单中声明广播接收器,系统会在广播发送时启动你的应用程序(如果应用程序尚未运行)。
如果你的应用程序的目标 API 级别为 26 或更高,则不能使用清单来声明隐式广播的接收器,一些免受限制的隐式广播除外。
1.1 在 manifest 中指定 <receiver> 元素。
<receiver android:name=".MyBroadcastReceiver" android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
<action android:name="android.intent.action.INPUT_METHOD_CHANGED" />
</intent-filter>
</receiver>
<intent filter>
中指定你的接收器订阅的广播操作。
1.2 创建 BroadcastReceiver 子类以及实现 onReceive(Context, Intent).
public class MyBroadcastReceiver extends BroadcastReceiver {
private static final String TAG = "MyBroadcastReceiver";
@Override
public void onReceive(Context context, Intent intent) {
StringBuilder sb = new StringBuilder();
sb.append("Action: " + intent.getAction() + "\n");
sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + "\n");
String log = sb.toString();
Log.d(TAG, log);
Toast.makeText(context, log, Toast.LENGTH_LONG).show();
}
}
系统包管理器在安装应用程序时注册接收器。接收器会成为你的应用程序的单独入口点,这意味着系统可以启动应用程序并传送广播,如果该应用程序并未正在运行。
系统会创建一个新的 BroadcastReceiver 组件对象来处理它收到的每个广播。此对象仅在调用 onReceive(Context, Intent) 期间有效。一旦你的代码从此方法返回,系统会认为该组件不再处于活动状态。
2. Context-registered receivers 上下文注册接收器
要使用上下文注册接收器,请执行以下步骤:
2.1 创建一个 BroadcastReceiver 实例。
BroadcastReceiver br = new MyBroadcastReceiver();
2.2 创建一个 IntentFilter 并通过调用 registerReceiver(BroadcastReceiver, IntentFilter) 注册接收器。
IntentFilter filter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION);
filter.addAction(Intent.ACTION_AIRPLANE_MODE_CHANGED);
this.registerReceiver(br, filter);
注意:要注册本地广播,改为调用
LocalBroadcastManager.registerReceiver(BroadcastReceiver, IntentFilter)
。
只要注册的 context 有效,接收器就会接收广播。例如:如果你使用 Activity Context 注册 ,只要 Activity 没有被销毁,你就会收到广播。如果你使用 Application Context 注册,只要应用程序正在运行,就会收到广播。
要停止接收广播,只需调用 unregisterReceiver(android.content.BroadcastReceiver)
。当你不再需要接收器或 context 不再有效时,请务必注销接收器。
留意你注册和注销接收者的位置,例如,如果你在 onCreate(Bundle) 注册,那你应该在 onDestroy() 中取消注册,以防止内存泄漏。如果你在 onResume() 中注册,那你应该在 onPause() 中取消注册,以防止多次注册(如果你不想在 pause 时收到广播,这可以减少不必要的系统开销)。不要在 onSaveInstanceState(Bundle) 中取消注册,因为该方法不会在用户移回到历史堆栈时调用它。
3. Effects on process state 对进程状态的影响
BroadcastReceiver(无论是否在运行)的状态会影响其所属进程的状态,从而影响其被系统杀死的可能性。例如,当一个进程执行一个 receiver(也就是说,onReceive() 方法被调用)时,它被认为是一个前台进程。除非存在极大的内存压力,否则系统会保持该进程。
但是,一旦代码返回 onReceive(),BroadcastReceiver 不再处于活动状态。系统会将接收器所在的进程视为低优先级进程,并可能将其终止,为其他更重要的进程提供资源。
出于这个原因,不应该从广播接收器中开启长时间运行的后台线程。onReceive() 方法完成之后,系统可以随时终止进程以回收内存,并且这样做会终止进程中运行的衍生线程。为了避免这种情况,你应该调用 goAsync()(如果你希望有更多时间在后台线程中处理广播),或者使用 JobScheduler 从 receiver 中安排一个 JobService,那么系统就知道该进程将继续执行工作。
下面的代码片段展示了一个 BroadcastReceiver 使用 goAsync() 来标志它需要更多的时间来结束当 onReceive() 完成之后。这使得它更适合执行后台线程。
public class MyBroadcastReceiver extends BroadcastReceiver {
private static final String TAG = "MyBroadcastReceiver";
@Override
public void onReceive(final Context context, final Intent intent) {
final PendingResult pendingResult = goAsync();
AsyncTask<String, Integer, String> asyncTask = new AsyncTask<String, Integer, String>() {
@Override
protected String doInBackground(String... params) {
StringBuilder sb = new StringBuilder();
sb.append("Action: " + intent.getAction() + "\n");
sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + "\n");
String log = sb.toString();
Log.d(TAG, log);
// Must call finish() so the BroadcastReceiver can be recycled.
pendingResult.finish();
return log;
}
};
asyncTask.execute();
}
}