全面总结之 BroadcastReceiver 篇

我们常常说 Android 四大组件:活动、服务、内容提供者、广播,前三个我们都说过了,开始说一下广播。

广播作为 Android 组件间的通信方式,可以非常方便的实现系统中不同组件之间的通信(甚至是不同应用之间的通信)。

在讲解内容提供者的时候,整个内容提供体系是由好几部分组成的:ContentProvider 用来提供数据,ContentResolver 用来操作数据,ContentObserver 用来监听数据变化。广播体系也是由几个部分组成的:广播发送方和广播接收方。

广播机制从发送接收方面可以分为:

  • 发送广播

  • 接收广播

从接收机制来讲可以分为:

  • 有序广播,广播接收器按照优先级接收广播,在传播途中可以被终止,终止之后,广播无法继续。

  • 无序广播,发送的广播会被每个注册的接收器接收到,无法被终止。

从接收范围可以分为:

  • 本地广播,发送的广播只能在程序内部传递,别的 App 无法接收到。

  • 全局广播,广播可以被所有注册过的 App 接收。

接收广播

我们想要接收广播(这个广播有可能来自于系统,来自于其他 App,也有可能是来自自己本身的 App),需要创建我们自己的广播接收器(BroadcastReceiver),创建接收器很简单,只需要两步:

  1. 创建我们的接收器
    创建接收器很简单,只需要让我们的类去继承 BroadcastReceiver 并且重写去 onReceive 方法,在这个方法中去执行我们接收到广播之后需要执行的内容。

    例如:

public class FirstReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Log.d("TTT", "FirstReceiver 接收到了,name = " + intent.getStringExtra("name"));
    }
}
  1. 在系统中注册该接收器,只有在系统中注册过的接收器才能够接收到广播。

注册接收器有两种方法:

  • 直接在代码中注册,调用 Context 的 registerReceiver 方法即可,该方法接收两个参数,一个是我们的 Receiver 对象,一个是指定了我们接收哪个广播的 IntentFilter 对象。这种方法被称为动态注册。动态注册只有当我们执行了那一段注册代码(也就是运行了 App 才会被注册),动态注册的接收器会存储在一个全局的静态变量中。
        IntentFilter filter = new IntentFilter("com.li_xyz.broadcasttest.my_receiver");
        FirstReceiver receiver = new FirstReceiver();
        registerReceiver(receiver,filter);
  • 在清单文件中进行注册,和 Activity、Service 和 ContentProvider 一样,BroadcastReceiver 同样需要在清单文件中注册,在注册的同时,我们还需要为该 BroadcastReceiver 执行他能够响应的广播,方法很简单,只需要在其 intent-filter 中指定一个 action 即可。这种方法被称为静态注册。当手机重启或者安装了新的 App 之后,系统会扫描手机中所有 App 的清单文件,并将其中的广播信息解析出来,存储在一个全局静态变量中。
        <receiver android:name=".FirstReceiver">
            <intent-filter>
                <action android:name="com.li_xyz.broadcasttest.my_receiver" />
            </intent-filter>
        </receiver>

需要注意的是,动态注册的接收器,需要我们的程序启动之后,才可以接收器才会工作,才可以接收到广播,原理也很容易理解,你的代码没有执行,所以自然也就没有注册了。
静态注册的代码则不需要启动程序,因为当系统启动(或者安装了新应用),系统会扫描手机中已经安装应用,将他们清单文件中的信息解析出来,所以自然也就不需要我们再去启动 App 了。

接收器的优先级

我们可以为我们的广播接收器设置一个优先级,优先级的取值范围是 -1000 ~ 1000,参数值越大,优先级越高,越先接收到广播(事实上,我们可以将优先级设置为比 1000 更大的数值,-1000 ~ 1000 只是官方文档中的数值)。

设置优先级的方式有两种:

  • 动态注册的接收器,调用 IntentFilter 对象的 setPriority 方法设置优先级。

  • 静态注册的接收器,在 <intent-filter> 元素中设置其 android:priority 属性。

对于普通广播来说,接收器的接收顺序遵循以下几个规则:

  1. 不管优先级高低,动态注册要先于静态注册接收。

  2. 同为静态或者同为动态注册的接收器,优先级高的要先接收。

  3. 对于静态广播器来说,如果优先级相同,先扫描的要优先于后扫描的接收。

  4. 对于动态广播器来说,如果优先级相同,先注册的要优先于后注册的接收。

对于有序广播来说,接收器的接收顺序规训下面的规则:

  1. 优先按照优先级接收,不管是动态注册还是静态注册。

  2. 同优先级的接收器,动态注册的要优先于静态注册的接收。

  3. 对于静态广播器来说,如果优先级相同,先扫描的要优先于后扫描的接收。

  4. 对于动态广播器来说,如果优先级相同,先注册的要优先于后注册的接收。

需要注意的是,这个排序并不是一直不变的,当程序被杀死之后, 已注册的动态广播接收器被从静态变量中移出,直到下次程序启动,再进行动态广播的注册,当然这里面的顺序也已经变更了一次。

截断广播

在广播接收器中执行 abortBroadcast() 方法,就可以将该广播截断,后面的广播接收器将无法再收到这条广播。但只有有序广播才可以被截断。

发送广播

有两个方法可以发送广播:

  • sendBroadcast(Intent intent):使用该方法发送普通广播,该方法接收一个 Intent 对象参数,通过该 Intent 标识要发送的广播。
Intent intent = new Intent("com.lixyz.mybroadcast.CUSTOM_BROADCAST");
sendBroadcast(intent);
  • sendOrderedBroadcast(Intent intent, String receiverPermission):使用这个方法发送有序广播,该方法接收两个参数,Intent 就不用说了,和普通广播一样。第二个参数是一个权限,使用一个字符串来设置接收方必须拥有的权限才能接收广播。 如果为空,则不需要权限。
<permission android:name="com.lixyz.broadcast.perssion" android:protectionLevel="normal"></permission>   
<uses-permission  android:name="com.lixyz.broadcast.perssion"/>  

```java
Intent intent = new Intent("com.lixyz.mybroadcast.CUSTOM_BROADCAST");
sendOrderedBroadcast(intent, "com.lixyz.broadcast.perssion");


发送广播就是这么简单。 ---- ### 本地广播 前面我们讲的都是全局广播,也就是说,只要其他 App 注册了我们的广播,就可以接收到我们发送的广播。这样就会引发安全性问题,比如说我们的广播携带一些关键性数据,如果被其他应用截获的话,就会造成数据泄漏。或者其他 App 不断的向我们的接收器发送各种垃圾广播的话,我们的接收器也就必须做出响应。 Android 引入了一套本地广播机制来帮助我们解决这个问题,本地广播只能够在我们 App 的内部传播,并且接收器也只能接收来自自身 App 发送的广播,这样安全性问题就不存在了。 本地广播的用法并不复杂,主要是使用了一个 LocalBroadcastManager 来对广播进行管理,并提供了发送广播和注册广播接收器的方法。 通过 LocalBroadcastManager 的 getInstance() 方法得到一个一个 LocalBroadcastManager 对象,在发送广播的时候调用的是 LocalBroadcastManager 对象的 sendBroadcast() 方法,仅此而已。 通过调用 LocalBroadcastManager 的 registerReceiver 方法来注册接收器,需要注意的是,本地广播只能动态注册,无法在清单文件中静态注册。 看代码: ```java public class MainActivity extends Activity { private IntentFilter intentFilter; private LocalReceiver localReceiver; private LocalBroadcastManager localBroadcastManager; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); localBroadcastManager = LocalBroadcastManager.getInstance(this); //获取实例 Button button = (Button)findViewById(R.id.button); button.setOnClickListener(new OnClickListener(){ @Override public void onClick(View v) { Intent intent = new Intent("com.example.broadcasttest.LOCAL_BROADCAST"); localBroadcastManager.sendBroadcast(intent); //发送本地广播 } }); intentFilter = new IntentFilter(); intentFilter.addAction("com.example.broadcasttest.LOCAL_BROADCAST"); localReceiver = new LocalReceiver(); localBroadcastManager.registerReceiver(localReceiver, intentFilter); //注册本地广播监听器 } @Override protected void onDestroy(){ super.onDestroy(); localBroadcastManager.unregisterReceiver(localReceiver); } class LocalReveiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { Toast.makeText(context, "received local broadcast", Toast.LENGTH_SHORT).show(); } } }

常用广播

关闭或打开飞行模式时的广播
Intent.ACTION_AIRPLANE_M;

充电状态,或者电池的电量发生变化; 电池的充电状态、电荷级别改变,不能通过组建声;
Intent.ACTION_BATTERY_CH;

表示电池电量低
Intent.ACTION_BATTERY_LO;

表示电池电量充足
Intent.ACTION_BATTERY_OK;

关闭或打开飞行模式时的广播
Intent.ACTION_AIRPLANE_MODE_CHANGED;

充电状态,或者电池的电量发生变化电池的充电状态、电荷级别改变,不能通过组建声明接收这个广播,只有通过Context.registerReceiver()注册
Intent.ACTION_BATTERY_CHANGED;

表示电池电量低
Intent.ACTION_BATTERY_LOW;

表示电池电量充足,即从电池电量低变化到饱满时会发出广播
Intent.ACTION_BATTERY_OKAY;

在系统启动完成后,这个动作被广播一次(只有一次)。
Intent.ACTION_BOOT_COMPLETED;

按下照相时的拍照按键(硬件按键)时发出的广播
Intent.ACTION_CAMERA_BUTTON;

当屏幕超时进行锁屏时,当用户按下电源按钮,长按或短按(不管有没跳出话框),进行锁屏时,android系统都会广播此Action消息
Intent.ACTION_CLOSE_SYSTEM_DIALOGS;

设备当前设置被改变时发出的广播(包括的改变:界面语言,设备方向,等,请参考Configuration.java)
Intent.ACTION_CONFIGURATION_CHANGED;

设备日期发生改变时会发出此广播
Intent.ACTION_DATE_CHANGED;

设备内存不足时发出的广播,此广播只能由系统使用,其它APP不可用
Intent.ACTION_DEVICE_STORAGE_LOW;

设备内存从不足到充足时发出的广播,此广播只能由系统使用,其它APP不可用
Intent.ACTION_DEVICE_STORAGE_OK;

发出此广播的地方frameworks\base\services\java\com\android\server\DockObserver.java
Intent.ACTION_DOCK_EVENT;

移动APP完成之后,发出的广播(移动是指:APP2SD)
Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE;

正在移动APP时,发出的广播(移动是指:APP2SD)
Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE;

Gtalk已建立连接时发出的广播
Intent.ACTION_GTALK_SERVICE_CONNECTED;

Gtalk已断开连接时发出的广播
Intent.ACTION_GTALK_SERVICE_DISCONNECTED;

在耳机口上插入耳机时发出的广播
Intent.ACTION_HEADSET_PLUG;

改变输入法时发出的广播
Intent.ACTION_INPUT_METHOD_CHANGED;

设备当前区域设置已更改时发出的广播
Intent.ACTION_LOCALE_CHANGED;

表示用户和包管理所承认的低内存状态通知应该开始。
Intent.ACTION_MANAGE_PACKAGE_STORAGE;

未正确移除SD卡(正确移除SD卡的方法:设置--SD卡和设备内存--卸载SD卡),但已把SD卡取出来时发出的广播 ,扩展介质(扩展卡)已经从 SD 卡插槽拔出,但是挂载点 (mount point) 还没解除 (unmount)
Intent.ACTION_MEDIA_BAD_REMOVAL;

按下"Media Button" 按键时发出的广播,假如有"Media Button" 按键的话(硬件按键)
Intent.ACTION_MEDIA_BUTTON;

插入外部储存装置,比如SD卡时,系统会检验SD卡,此时发出的广播?
Intent.ACTION_MEDIA_CHECKING;

已拔掉外部大容量储存设备发出的广播(比如SD卡,或移动硬盘),不管有没有正确卸载都会发出此广播, 用户想要移除扩展介质(拔掉扩展卡)。
Intent.ACTION_MEDIA_EJECT;

插入SD卡并且已正确安装(识别)时发出的广播, 扩展介质被插入,而且已经被挂载。
Intent.ACTION_MEDIA_MOUNTED;

拓展介质存在,但使用不兼容FS(或为空)的路径安装点检查介质包含在Intent.mData领域。
Intent.ACTION_MEDIA_NOFS;

外部储存设备已被移除,不管有没正确卸载,都会发出此广播, 扩展介质被移除。
Intent.ACTION_MEDIA_REMOVED;

广播:已经扫描完介质的一个目录
Intent.ACTION_MEDIA_SCANNER_FINISHED;

请求媒体扫描仪扫描文件并将其添加到媒体数据库。
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;

广播:开始扫描介质的一个目录
Intent.ACTION_MEDIA_SCANNER_STARTED;

广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。
Intent.ACTION_MEDIA_SHARED; Intent.ACTION_MEDIA_UNMOUNTABLE;

广播:扩展介质存在,但是还没有被挂载 (mount)
Intent.ACTION_MEDIA_UNMOUNTED Intent.ACTION_NEW_OUTGOING_CALL;

成功的安装APK之后、设备上新安装了一个应用程序包。、一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_ADDED;

一个已存在的应用程序包已经改变,包括包名
Intent.ACTION_PACKAGE_CHANGED;

清除一个应用程序的数据时发出的广播(在设置--应用管理--选中某个应用,之后点清除数据时?)用户已经清除一个包的数据,包括包名(清除包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_DATA_CLEARED;

触发一个下载并且完成安装时发出的广播,比如在电子市场里下载应用?
Intent.ACTION_PACKAGE_INSTALL;

成功的删除某个APK之后发出的广播, 一个已存在的应用程序包已经从设备上移除,包括包名(正在被安装的包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_REMOVED;

替换一个现有的安装包时发出的广播(不管现在安装的APP比之前的新还是旧,都会发出此广播?)
Intent.ACTION_PACKAGE_REPLACED;

用户重新开始一个包,包的所有进程将被杀死,所有与其联系的运行时间状态应该被移除,包括包名(重新开始包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_RESTARTED;

插上外部电源时发出的广播
Intent.ACTION_POWER_CONNECTED;

已断开外部电源连接时发出的广播
Intent.ACTION_POWER_DISCONNECTED; Intent.ACTION_PROVIDER_CHANGED;

重启设备时的广播
Intent.ACTION_REBOOT;

屏幕被关闭之后的广播
Intent.ACTION_SCREEN_OFF;

屏幕被打开之后的广播
Intent.ACTION_SCREEN_ON;

关闭系统时发出的广播
Intent.ACTION_SHUTDOWN;

时区发生改变时发出的广播
Intent.ACTION_TIMEZONE_CHANGED;

时间被设置时发出的广播
Intent.ACTION_TIME_CHANGED;

广播:当前时间已经变化(正常的时间流逝), 当前时间改变,每分钟都发送,不能通过组件声明来接收
,只有通过Context.registerReceiver()方法来注册
Intent.ACTION_TIME_TICK;

一个用户ID已经从系统中移除发出的广播
Intent.ACTION_UID_REMOVED;

设备已进入USB大容量储存状态时发出的广播?
Intent.ACTION_UMS_CONNECTED;

设备已从USB大容量储存状态转为正常状态时发出的广播?
Intent.ACTION_UMS_DISCONNECTED; Intent.ACTION_USER_PRESENT;

设备墙纸已改变时发出的广播
Intent.ACTION_WALLPAPER_CHANGED;

参考

Copyright© 2020-2022 li-xyz 冀ICP备2022001112号-1