四 boardcastreceiver 面试详解

来源:互联网 发布:psv重构数据库365 编辑:程序博客网 时间:2024/05/29 18:44

一 BoardcastReceiver 定义

BoardcastReceiver 中文译为广播接收器,贵为 Android 四大组件之一。以观察者模式实现的一个全局的监听器。分为两个角色:广播发送者和广播接受者。

二 BoardcastReceiver 作用

01 用于监听/接收 应用发出的广播消息,并作出反应。

02 分应用场景分

1 不同组件之间的通信(包括同一应用间和不同应用间)

2 与 Android 系统在特定情况下的通信,如:电话呼入时,短信接收时,网络连接时。

3 多线程通信。

三 BoardcastReceiver 实现原理

广播使用了观察者模式,基于消息的发布/订阅事件模型,将发送者和接收者进行解耦。
在这个消息发布/订阅事件模型中有三个角色:
image

原理描述:
1 广播接收者通过 Binder 机制在 AMS 注册
2 广播发送者通过 Binder 机制向 AMS 发送广播
3 AMS 根据广播发送者要求(IntentFilter 和 Permission),在已注册列表中,寻找合适的广播接收者
4 AMS 将广播发送到合适的广播接收者相应的消息循环队列中
5 广播接收者通过消息循环拿到此广播,并回调 onReceive()

四 BoardcastReceiver 使用

01 自定义广播接收者 BoardcastReceiver

  • 继承自BroadcastReceivre基类
  • 必须复写抽象方法onReceive()方法
    1. 广播接收器接收到相应广播后,会自动回调onReceive()方法
    2. 一般情况下,onReceive方法会涉及与其他组件之间的交互,如发送Notification、启动service等
    3. 默认情况下,广播接收器运行在UI线程,因此,onReceive方法不能执行耗时操作,否则将导致ANR。

02 注册广播接收器

1 静态注册

  • 在AndroidManifest.xml里通过标签声明, 当此App首次启动时,系统会自动实例化mBroadcastReceiver类,并注册到系统中
实例代码:
<receiver     //这个接收器的开关    android:enabled=["true" | "false"]    //此broadcastReceiver能否接收其他App的发出的广播    //默认值是由receiver中有无intent-filter决定的:如果有intent-filter,默认值为true,否则为false    android:exported=["true" | "false"]    android:icon="drawable resource"    android:label="string resource"    //继承BroadcastReceiver子类的类名    android:name=".mBroadcastReceiver"    //具有相应权限的广播发送者发送的广播才能被此BroadcastReceiver所接收;    android:permission="string"   //BroadcastReceiver运行所处的进程   //默认为app的进程,可以指定独立的进程   //注:Android四大基本组件都可以通过此属性指定自己的独立进程    android:process="string" >   //用于指定此广播接收器将接收的广播类型   //本示例中给出的是用于接收网络状态改变时发出的广播   <intent-filter>   <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />   </intent-filter></receiver>

2 动态注册

  • 在代码中通过调用Context的registerReceiver()方法进行动态注册BroadcastReceiver
实例代码:
@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); }//注册广播后,要在相应位置记得销毁广播//即在onPause() 中unregisterReceiver(mBroadcastReceiver)//当此Activity实例化时,会动态将MyBroadcastReceiver注册到系统中//当此Activity销毁时,动态注册的MyBroadcastReceiver将不再接收到相应的广播。 @Override protected void onPause() {     super.onPause();      //销毁在onResume()方法中的广播     unregisterReceiver(mBroadcastReceiver);     }}

注意:

动态广播最好在Activity的onResume()注册、onPause()注销。

* 原因:*

1 对于动态广播,有注册就必然得有注销,否则会导致内存泄露
2 在onResume()注册、onPause()注销是因为onPause()在App死亡前一定会被执行,从而保证广播在App死亡前一定会被注销,从而防止内存泄露。

03 广播发送者向AMS发送广播

1 广播的发送

  • 广播是用”意图(Intent)“标识
  • 定义广播的本质:定义广播所具备的“意图(Intent)”
  • 广播发送:广播发送者将此广播的”意图“通过sendBroadcast()方法发送出去

2 广播的类型

1. 普通广播(Normal Broadcast)

即开发者自身定义intent的广播(最常用)。发送广播使用如下:

Intent intent = new Intent();//对应BroadcastReceiver中intentFilter的actionintent.setAction(BROADCAST_ACTION);//发送广播sendBroadcast(intent);

若被注册了的广播接收者中注册时intentFilter的action与上述匹配,则会接收此广播(即进行回调onReceive())。如下mBroadcastReceiver则会接收上述广播,若发送广播有相应权限,那么广播接收者也需要相应权限

2. 系统广播(System Broadcast)

  • Android中内置了多个系统广播:只要涉及到手机的基本操作(如开机、网络状态变化、拍照等等),都会发出相应的广播
  • 每个广播都有特定的Intent - Filter(包括具体的action)。
  • 当使用系统广播时,只需要在注册广播接收者时定义相关的action即可,并不需要手动发送广播,当系统有相关操作时会自动进行系统广播

3. 有序广播(Ordered Broadcast)

4. App应用内广播(Local Broadcast)

  • 背景
    • Android中的广播可以跨App直接通信(exported对于有intent-filter情况下默认值为true)
  • 冲突
    • 其他App针对性发出与当前App intent-filter相匹配的广播,由此导致当前App不断接收广播并处理;
    • 其他App注册与当前App一致的intent-filter用于接收广播,获取广播具体信息;
      即会出现安全性 & 效率性的问题。
  • 解决方案:使用App应用内广播(Local Broadcast)
    • App应用内广播可理解为一种局部广播,广播的发送者和接收者都同属于一个App。
    • 相比于全局广播(普通广播),App应用内广播优势体现在:安全性高 & 效率高
  • 使用

    • 将全局广播设置成局部广播

    1 注册广播时将exported属性设置为false,使得非本App内部发出的此广播不被接收;
    2 在广播发送和接收时,增设相应权限permission,用于权限验证;
    3 发送广播时指定该广播接收器所在的包名,此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中。

    • 使用封装好的LocalBroadcastManager类,该类是一个单例模式的类:实现原理是 handler 消息机制。

    实例代码:

    //注册应用内广播接收器//步骤1:实例化BroadcastReceiver子类 & IntentFilter mBroadcastReceiver mBroadcastReceiver = new mBroadcastReceiver(); IntentFilter intentFilter = new IntentFilter(); //步骤2:实例化LocalBroadcastManager的实例localBroadcastManager = LocalBroadcastManager.getInstance(this);//步骤3:设置接收广播的类型 intentFilter.addAction(android.net.conn.CONNECTIVITY_CHANGE);//步骤4:调用LocalBroadcastManager单一实例的registerReceiver()方法进行动态注册 localBroadcastManager.registerReceiver(mBroadcastReceiver, intentFilter);//取消注册应用内广播接收器localBroadcastManager.unregisterReceiver(mBroadcastReceiver);//发送应用内广播Intent intent = new Intent();intent.setAction(BROADCAST_ACTION);localBroadcastManager.sendBroadcast(intent);

5 粘性广播(Sticky Broadcast)

五 BoardcastReceiver 注意

01 动态注册和静态注册的区别

image

02 onReceiver 方法中返回 context 的区别

  • 对于静态注册(全局+应用内广播),回调onReceive(context, intent)中的context返回值是:ReceiverRestrictedContext;
  • 对于全局广播的动态注册,回调onReceive(context, intent)中的context返回值是:Activity Context;
  • 对于应用内广播的动态注册(LocalBroadcastManager方式),回调onReceive(context, intent)中的context返回值是:Application Context。
  • 对于应用内广播的动态注册(非LocalBroadcastManager方式),回调onReceive(context, intent)中的context返回值是:Activity Context;

六 感谢

本文整理的内容大部分来自于:

作者:Carson_Ho
链接:http://www.jianshu.com/p/ca3d87a4cdf3
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。