Broadcast 知识梳理(1) - BroadcastReceiver 基本概念

一、基本概念

1.1 实现广播接收者

首先,我们需要创建一个广播接收者,继承于BroadcastReceiver并重写它的onReceive方法。

public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {}
}

在创建完接收者之后,还需要进行注册,告诉系统有这个监听者。广播注册的方式分为:静态注册和动态注册。

静态注册

静态注册在AndroidManifest.xml中进行指定。

<receiver android:enabled=["true" | "false"]
    android:exported=["true" | "false"]
    android:icon="drawable resource"
    android:label="string resource"
    android:name="string"
    android:permission="string"
    android:process="string" >
. . .
</receiver>
  • exportedexported决定了 广播接收者所能收到广播的范围,假如为false,那么只有同一个App,或者userId相同的App发出的广播它才能够收到,并不是指同一个进程。exported一般情况下默认为false,唯一例外的是假如设置了intent-filter,那么默认值为true。对于静态注册的广播,在Android 3.1之后,应用如果没有启动并且Intent中包含了FLAG_INCLUDE_STOPPED_PACKAGES属性,那么会先调起应用,否则在应用没有启动的情况下将无法收到广播。

  • permission:如果设置了permission,那么只有 具有相应权限的广播发送方 发送的广播才能被此BroadcastReceiver接收。Android广播的权限机制是双向的,即我们既可以 要求发送者具有权限,也可以 要求接收者具有权限,才能完成端到端的通信过程,这里就是 要求发送者具有权限

  • process:运行所处的进程,默认为App的进程。

动态注册

动态注册的广播无需在AndroidManifest.xml进行声明,在代码中进行注册和注销即可。

//注册广播。
registerReceiver(BroadcastReceiver receiver, IntentFilter filter)
//注销广播。
unregisterReceiver(BroadcastReceiver receiver);

1.2 发送广播

广播的发送者通过Intent将其意图发送出去,系统找到匹配的接收者,发送广播的一般方式如下:

Intent intent = new Intent();
intent.setAction(INTENT_ACTION);
sendBroadcast(intent);

1.3 广播类型

广播可以分为以下几类:无序广播、有序广播和粘性广播。

1.3.1 无序广播

无序广播指的是所有广播接收者收到广播的顺序是没有规律的。

1.3.2 有序广播

有序广播指的是发送出的广播被BroadcastReceiver按照priority从大到小的顺序接收,当priority相同时,动态广播优先静态广播,发送有序广播的方式为:

sendOrderedBroadcast(intent, receiverPermission, ...)

对于有序广播有一个特点,先接收的BroadcastReceiver具有拦截广播的权利,拦截的方法为onReceive方法中调用abortBroadcast()方法。

1.3.3 粘性广播

已经废弃,它是用来处理先收到广播然后才注册的情况。

1.4 应用内广播

假如exported属性为true,那么是允许两个不同应用通过广播进行通信的。就可能出现 安全隐患

  • 其他App可能会针对性地发出与当前Appintent-filter相匹配的广播,导致当前App不断接收到广播并处理。
  • 其他App可以注册与当前App相匹配的intent-filter,从而获取广播具体信息。

为了避免出现以上的安全问题,有以下的解决方法:

  • 设置exported属性为false
  • 设置权限。
  • 发送广播时,指定具体的包名。

假如我们的广播只需要在应用内部通信,那么可以采用封装好的LocalBroadcastManager类,用于解决安全问题。

传统的广播是通过Binder来实现的,而LocalBroadcastManager则是采用Handler的方式。当注册广播的时候,其实将Receiver添加到单例对象LocalBroadcastManager维护的列表当中,发送消息的时候,通过Receiver所关联的action找到它,最后回调它的onReceive方法。

因此,只有通过LocalBroadcastManager注册的BroadcastReceiver才能收到通过LocalBroadcastManager发出的广播。

具体的代码实现可以参考这篇文章 LocalBroadcastManager 的实现原理,还是 Binder?

二、一些需要注意的点

2.1 权限问题

通过权限可以也可以解决我们之前谈到的安全问题,Broadcast的权限是双向的。

2.1.1 要求接收者具有权限

这种方式 用于防止广播信息泄露

发送者AndroidManifest.xml中定义权限。

<permission android:name = "com.android.permission.RECV_XXX"/>

发送者在发送广播的时候,采用带有权限的接口进行发送。

sendBroadcast("com.android.XXX_ACTION", "com.android.permission.RECV_XXX");


接收者 如果希望能收到广播,那么需要在它的AndroidManifest.xml进行声明使用该权限。

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

2.1.2 要求发送者具有权限

这种方式 用于防止外部应用恶意地发送广播,导致接收者一直在处理

接收者AndroidManifest.xml中定义权限。

<permission android:name="com.android.SEND_XXX"/>

在接收者的AndroidManifest.xml声明BroadcastReceiver的时候,通过permission字段指定权限。

<receiver android:name=".XXXReceiver" 
          android:permission="com.android.permission.SEND_XXX"> 
    <intent-filter>
         <action android:name="com.android.XXX_ACTION" /> 
    </intent-filter>
</receiver>


发送者 如果希望广播能被该接收者收到,那么需要在AndroidManifest.xml中声明使用该权限。

<uses-permission android:name="com.android.permission.SEND_XXX“></users-permission>

2.2 ANR

BroadcastReceiver方法中,不要进行耗时的操作,超过10s会发生BroadcastQueue TimeoutANR异常。

2.3 onReceive 传入的 Context

  • 静态注册的BroadcastReceiver,其Contextandroid.app.ReceiverRestrictedContext

  • 动态注册的普通BroadcastReceiver,与调用registerReceiver方法的Context有关,如果是通过Application Context注册的,那么ContextApplication Context,如果是Activity Context,那么其ContextActivity Context

  • LocalBroadcastManager动态注册的BroadcastReceiver,其ContextApplication Context

三、参考文章

Android Broadcast 和 BroadcastReceiver 的权限限制
Android 总结篇系列:Android 广播机制
LocalBroadcastManager 的实现原理,还是 Binder?

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

推荐阅读更多精彩内容

  • 【Android 广播】 BroadcastReceiver简介 BroadcastReceiver(广播接收器)...
    Rtia阅读 3,337评论 1 17
  • 本文转载自http://www.cnblogs.com/lwbqqyumidi/p/4168017.html 1....
    Ernest_Chang阅读 684评论 0 3
  • 广播接收器 是Android的四大组件之一,可见广播在Android中的重要性; 1. 什么是广播? 广播(Bro...
    Lost_Robot阅读 1,807评论 2 10
  • 前言 本来想写一下广播的,发现查阅后有整理的不错的,只好转载图个简便,日后好复习转载:http://www.cnb...
    提升即效率阅读 1,362评论 0 10
  • 01出发和归途 他们在大雪纷飞的日子出发,在另一个大雪纷飞的日子归来。 途中有新生命的诞生,终点有年老者安息归去。...
    Arya周阅读 681评论 0 1