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 中指定 <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();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容