Android广播的那些事儿

Android App可以接收来自系统和其他App的广播消息,也可以向它们发送广播消息,比较类似于“发布-订阅”的设计模式,本文主要介绍广播的类型,如何注册广播,如何发送广播以及使用广播需要注意的一些事儿。

I. 广播的分类

  1. 无序广播
    没有顺序的广播,广播的接收方没有严格的顺序可言,不可中断。

  2. 有序广播
    在注册时可指定优先级,优先级高的广播接收者优先收到广播,优先级以一个整数来标识,数值越大优先级越高。可中断,可再修饰。

  3. 粘滞广播
    发出的广播会滞留,注册时间可晚于发送时间,其他功能与无序广播相同。

II. 注册广播

编写一个广播接收者,通常是继承BroadcastReceiver并重写onReceive(Context,Intent)方法。

public class MyReceiver extends BroadcastReceiver {
    
    private static final String TAG = "MyReceiver";
    
    @Override
    public void onReceive(Context context, Intent intent) {
        // TODO: do somethings
        // 这个方法运行在主线程下
        String action = intent.getAction();
        Log.d(TAG, "onReceive: " + action);
    }
}

编写完广播接收者后,就需要进行注册(订阅),告诉系统这个广播接收者对哪些广播感兴趣。注册的方式有静态注册和动态注册两种:

  • 在AndroidManifest.xml文件中声明(静态注册)
<receiver android:name=".MyReceiver">
    <intent-filter>
        <action android:name="android.intent.action.BOOT_COMPLETED"/>
        <action android:name="jdqm.intent.action.TEST"/>
    </intent-filter>
</receiver>
  • 通过Java代码注册(动态注册)
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("android.intent.action.BOOT_COMPLETED");
intentFilter.addAction("jdqm.intent.action.TEST");
registerReceiver(myReceiver, intentFilter);

通过动态注册的广播接收者,在宿主(注册时所使用的Context)的生命周期期间都是有效的。当然你也可以在适当的时间调用unregisterReceiver(BroadcastReceiver)来解除注册,这个“适当”取决于具体的业务需求。例如使用Activity的Context在onCreate(Bundle) 中注册的一个广播接收者,可以在onDestory()方法回调时解除注册来防止广播接收者泄漏。原则:不重复注册,不泄露。

以上注册的广播接收者对 android.intent.action.BOOT_COMPLETED 和 jdqm.intent.action.TEST 这两种action的广播感兴趣,后者是自定义的广播,前者是开机完成时由系统发出(通常自启动的应用会注册这个广播),但注册这个广播须要以下权限:

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>

III. 发送广播

  1. 发送无序广播
Intent intent = new Intent("jdqm.intent.action.TEST");
sendBroadcast(intent);
  1. 发送有序广播
Intent intent = new Intent("jdqm.intent.action.TEST");
//第二个参数是权限
sendOrderedBroadcast(intent, null);
  1. 发送本地广播
Intent intent = new Intent("jdqm.intent.action.TEST");
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);

本地广播只有本应用内通过LocalBroadcastManager.getInstance(this).registerReceiver方法注册的广播接收者能收到,具有更高的安全性,效率也更高(不用跨进程通信)。

  1. 发送粘滞广播
Intent intent = new Intent("jdqm.intent.action.TEST");
sendStickyBroadcast(intent);

这种类型广播在Android6.0中已经被标记被过时, 它有不安全(任何App都能访问), 没有保护 (任何App都能修改)等问题。另外发送这种广播需要以下权限

<uses-permission android:name="android.permission.BROADCAST_STICKY"/>

IV. 接收顺序

  1. 对于无序广播,动态注册的广播接收者会先收到,可以从源码中得到理论支撑。在BroadcastQueue类中有两个集合
//存储所有动态注册的无序广播接受者
final ArrayList<BroadcastRecord> mParallelBroadcasts = new ArrayList<>();

//存储所有静态注册和动态注册的有序广播接收者
final ArrayList<BroadcastRecord> mOrderedBroadcasts = new ArrayList<>();

然后在其processNextBroadcast(boolean fromMsg)方法中,首先是处理了mParallelBroadcasts集合。

  1. 对于有序广播,优先级高的接受者先收到,如果优先级相同,顺序就是不确定的。先收到的接收者可以调用abortBroadcast()来中断此广播,后续优先级较低的接受者将无法收到。除了中断还可以调用setResultXxx()方法来往广播添加数据,后续的接收者可以读取这些数据。

IV. 安全性与实践

  1. 如果你的广播不需要发送给本应用以外的组件,使用LocalBroadcastManager来发送广播,这样安全性和效率都比较高

  2. 静态注册有可能造成大量的App启动,这将会影响系统的性能,所以尽量使用动态注册来替代静态注册。这一点Android系统就做出了很好的示范,比如 CONNECTIVITY_ACTION 这个广播只发送给动态注册的广播接收者。

  3. 不在广播的Intent中包含敏感的信息,因为只要注册了这个广播就能读取到这些信息。你可以通过以下3中方式来获得一定的安全性。

  • 通过使用权限来发送广播,这样只有声明了该权限的应用才能收到广播。但是你很难确保你的权限不被泄漏。
  • Android4.0及以上版本,在发送广播的时候可以通过setPackage来指定package(可以指定多个),这样只有匹配的package能接受到。
  • 使用LocalBroadcastManager来发送本地广播。
  1. 当你注册了一个广播,意味着任何App都可以给你发送广播,以下有三点可以限制接收者:
  • 注册的时候增加权限。
  • 在AndroidManifest.xml注册receivers时,将android:exported属性设为false。
  • 使用LocalBroadcastManager来注册。
  1. action的命名空间是全局的,这意味着action有可能会与其他App冲突,所以最好是有一个自己的命名空间。

  2. 因为广播接收者是运行在主线程,它应该快速地被执行并且return,所以不要在onReveive方法中做比较耗时的操作。

  3. 不要在广播接收者中启动activitys,这违背了用户的使用习惯,特别是不止一个接收者时。这种情况下可以考虑展示一个notification来替代。

V. 其他

  1. 系统广播的action的完整列表在Android SDK下的 BROADCAST_ACTIONS.TXT,路径为: Android/sdk/platforms/android-26/data/BROADCAST_ACTIONS.TXT

  2. Android3.1开始,系统的package manager将记录处于停止状态的应用。默认情况下,静态注册了广播的处于“停止”状态的应用,是不会被启动的,即不会收到广播。当然你也可以为Intent指定flag来该变这个行为:

  • FLAG_INCLUDE_STOPPED_PACKAGES:包含处于“停止”状态的应用;
  • FLAG_EXCLUDE_STOPPED_PACKAGES:不包含处于“停止”状态的应用;

如果Intent不包含(或都包含)这两个flag,则表现形式是包含处于“停止”转态的应用,但是系统默认添加了FLAG_EXCLUDE_STOPPED_PACKAGES这个flag,这一点在源码中有所体现:

ActivityManagerService#broadcastIntentLocked

final int broadcastIntentLocked(ProcessRecord callerApp,
    String callerPackage, Intent intent, String resolvedType,
    IIntentReceiver resultTo, int resultCode, String resultData,
    Bundle resultExtras, String[] requiredPermissions, int appOp, Bundle bOptions,
    boolean ordered, boolean sticky, int callingPid, int callingUid, int userId) {
    
    intent = new Intent(intent);

    // By default broadcasts do not go to stopped apps.
    intent.addFlags(Intent.FLAG_EXCLUDE_STOPPED_PACKAGES);
    ...
}

这就意味着如果你想启动处于“停止”状态的应用,必须添加FLAG_INCLUDE_STOPPED_PACKAGES这个flag。那么一个应用在什么情况下会处于停止状态?①应用首次安装并且没有启动过;②被人为地强制停止。开机完成的广播就是FLAG_EXCLUDE_STOPPED_PACKAGES这种类型的Intent,这意味着如果你的应用被停止了,开机自启就会失效。下一篇文章将从源码的角度来分析广播的工作流程。

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

推荐阅读更多精彩内容