Android Broadcast 探索(一)— 概览

一、广播概述

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 中指定 元素。

    
        
        
    

中指定你的接收器订阅的广播操作。

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 asyncTask = new AsyncTask() {
            @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();
    }
}

你可能感兴趣的:(Android Broadcast 探索(一)— 概览)