1、广播实现原理

Android 中的广播使用了设计模式中的观察者模式:基于消息的发布/订阅事件模型。

模型中有3个角色:1. 消息订阅者(广播接收者) 2. 消息发布者(广播发布者) 3. 消息中心( AMS ,即 Activity Manager Service )

原理描述:

1. 广播接收者 通过 Binder 机制在 AMS 注册

2. 广播发送者 通过 Binder 机制向 AMS 发送广播

3. AMS 根据 广播发送者 要求,在已注册列表中,寻找合适的广播接收者

寻找依据:IntentFilter / Permission

4. AMS 将广播发送到合适的广播接收者相应的消息循环队列中;

5. 广播接收者通过 消息循环 拿到此广播,并回调 onReceive()

特别注意:广播发送者 和 广播接收者的执行 是 异步的,发出去的广播不会关心有无接收者接收,也不确定接收者到底是何时才能接收到;

2、广播的分类

  • 无序广播

也就是最常见的标准广播,通过SendBroadcast()方法发送。

  • 有序广播

针对广播接收方而言,通过sendOrderedBroadcast(intent)发送,发送出去的广播被广播接收者按照优先级先后顺序接收,相同优先级的动态注册的广播优先,每次只能有一个接受者收到,接受者收到广播后,可以通过setResultData来传递数据给下一个接收者,也可以通过abortBroadcast()来终止广播继续向下传递。

  • 粘性广播

调用SendStickyBroadcast()方法发送,需要android.Manifest.permission.BROADCAST_STICKT权限,注册者可以接受到注册广播前发送者发送的最后一次广播。目前API 21中已标记为Deprecated,不推荐使用。系统中电量的广播就是使用粘性广播发送的。

  • 本地广播

通过系统LocalBroadcastManager发送,只能在当前应用内接收。相对于其他类型广播而言,安全性高&效率高。本地广播只能通过LocalBroadcastManager动态注册。

  • 系统广播

有的地方把这个也算一个分类,这里也提一下,系统广播就是Android系统内置的广播,用来通知应用一些系统状态的改变,如:息屏亮屏,电量变化,网络状态变化。使用者只需注册对应的Action, 系统有相关操作时会自动广播。

3、广播注册的方式

  • 静态注册

在Manifest文件中,通过xml标签注册。

        

exported属性表示是否暴露给其他应用,设置为true, 则可以接收到其他应用发送的广播,默认值是由BroadcastReceiver中有无Intent-filter决定的,如果有Intent-filter,默认值为true,否则为false。

上面例子中,当此App首次启动时,系统会自动实例化XXXReceiver类,并注册到系统中。

  • 动态注册

在代码中通过调用Context的registerReceiver()方法进行动态注册

@Override protected void onResume() {  super.onResume();  //实例化BroadcastReceiver子类 & IntentFilter  mBroadcastReceiver mBroadcastReceiver = new mBroadcastReceiver();  IntentFilter intentFilter = new IntentFilter(); //设置接收广播的类型   intentFilter.addAction(android.net.conn.CONNECTIVITY_CHANGE) ;   //调用Context的registerReceiver()方法进行动态注册   registerReceiver(mBroadcastReceiver, intentFilter); }

动态注册时要注意注册与反注册成对出现,否则会造成内存泄漏。

4、两种广播注册方式的比较

5、LocalBroadcastManager实现原理

LocalBroadcastManager虽然使用和普通广播没有太大差别,但是原理却是完全不同。LocalBroadcastManager内部维护了mReceivers和mActions两个HashMap,

mReceivers 是接收器和IntentFilter的对应表,主要作用是方便在unregisterReceiver(…)取消注册,同时作为对象锁限制注册接收器、发送广播、取消接收器注册等几个过程的并发访问。

mActions 以Action为 key,注册这个Action的BroadcastReceiver链表为 value。mActions 的主要作用是方便在广播发送后快速得到可以接收它的BroadcastReceiver。

在注册广播时,其实是在更新这两个Map.

HashMap> mReceivers            = new HashMap>();HashMap> mActions            = new HashMap>();

发送广播时,根据Action从mActions中取出ReceiverRecord列表,找出action匹配的广播,然后通过Handler发送消息,在Handler的handleMessage中,取出匹配的广播列表,依次回调onReceive方法。

6、常见系统广播

从Android 7.0开始,系统不会再发送广播ACTION_NEW_PICTURE和ACTION_NEW_VIDEO,对于广播CONNECTIVITY_ACTION必须在代码中使用registerReceiver方法注册接收器,在AndroidManifest文件中声明接收器不起作用。

从Android 8.0开始,对于大多数系统隐式广播,不能在AndroidManifest文件中注册。

7、广播安全性

Android系统中的广播可以跨进程直接通信,会产生以下两个问题:

其他APP可以接收到当前APP发送的广播,导致数据外泄。

其他APP可以向当前APP放广播消息,导致APP被非法控制。

(1)发送广播

  • 发送广播时,增加相应的permission,用于权限验证。

  • 在Android 4.0及以上系统中发送广播时,可以使用setPackage()方法设置接受广播的包名。

  • 使用局部广播。

(2)接受广播

  • 注册广播接收器时,增加相应的permission,用于权限验证。

  • 注册广播接收器时,设置android:exported的值为false。

  • 使用局部广播。

发送广播时,如果增加了permission,那接受广播的APP必须申请相应权限,这样才能收到对应的广播,反之亦然。

8、​广播中能执行耗时操作吗?

不能,广播接收默认是在主线程中运行,在前面Android高频面试专题 - 进阶篇(一)ANR中讲过,广播超时是10s(前台)和60s(后台),如果耗时超过这个时间,就会抛出ANR,所以如果需要在广播内执行耗时操作,可以在​广播内启动一个IntentService来执行。​


更多完整面试专题和进阶知识分享,尽在“Android扫地僧”

更多相关文章

  1. Android(安卓)2.2 SDK操作系统界面截屏抢先看
  2. android手机文件后缀揭秘
  3. Android(安卓)进程保活招式大全
  4. Android(安卓)4.1.2系统添加重启功能
  5. Android系统给第三方app签名流程
  6. Android软硬整合技术(HAL&Framework):高阶的课程篇
  7. android 调用系统的照相机和图库实例详解
  8. Android(安卓)/system 目录解析
  9. 苹果MAC操作系统上搭建Android开发平台环境

随机推荐

  1. Android軟體《Time Lapse Creator》手機
  2. Android 时区的修改增删,如何在菜单里增加
  3. 程序员也是会浪漫的->打造浪漫的Android
  4. Android UI技巧(一)——Android中伸缩自如
  5. Android主题和样式
  6. android拖动条和评分组件的使用
  7. android sqlist中游标下标越界问题解决方
  8. Android简单的计步器应用实现
  9. 【Android 内存优化】Bitmap 内存缓存 (
  10. Android App 手機軟體20個私房推薦,系統工