0
点赞
收藏
分享

微信扫一扫

Android四大组件之BroadcastReceiver


1.定义

Android四大组件之一,没有可视化界面,BroadcastReceive也就是“广播接收者”的意思,顾名思义,就是用来接收来自系统和应用中的广播。

在Android系统中,广播体现在方方面面,例如当开机完成后系统会产生一条广播,接收到这条广播就能实现开机启动服务的功能,当网络状态改变时,系统会产生一条广播,接收到这条广播,就能及时的做出提示和保存数据等操作,当电池的电量改变的时候,系统会产生一条广播,接收到这条广播就能在电量低的时候告知用户,及时保存进度等等
Android中的广播机制设计的非常出色,很多事情原本需要开发者亲自操作的额,现在只需要广播告知自己就可以了,大大的减少了开发的工作量和开发周期,而作为应用开发者,就需要熟练掌握Android系统提供的一个开发利器,那就是BroadcastReceive。一个应用可以在发生特定事件的时候发送Broadcast,系统中任何应用只要注册对应的receiver,就会接收到此Broadcast.

用于不同组件和多线程之间的通信(含 :应用内 / 不同应用之间),Android系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件。
Android系统在运行的过程中,会产生很多事件,比如开机、电量改变、收发短信、拨打电话、屏幕解锁

广播有接收方和发送方,就算没有接收方,发送方也会发送数据。下面定义接收和发送代码

2.广播接收者BroadcastReceiver

2.1定义BroadcastReceiver

/**
* 定义一个广播接收者继承BroadcastReceivre基类
*/
public class TestBroadReceive extends BroadcastReceiver {
/**
* 复写onReceive,接收到广播自动执行此方法
* @param context
* @param intent
*/
@Override
public void onReceive(Context context, Intent intent) {
//写入接收到广播之后的操作,这里打印广播要发的信息
String message = intent.getStringExtra("message");
System.out.println(message+"==============================================");
}
}

2.2注册BroadcastReceiver

注册的方式分为两种:静态注册、动态注册

静态注册

  • 注册方式:在AndroidManifest.xml里通过<receive>标签声明
  • 属性说明:

<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>

  • 注册示例

<!--这里的广播接收器是TestBroadReceive-->
<receiver android:name="com.javayihao.demo.TestBroadReceive" >
<intent-filter>
<!--用于接收网络状态改变时发出的广播-->
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
<!--也可以是自定义广播,比如com.javayihao.demo.TestBroadReceive.action-->
<!--<action android:name="com.javayihao.demo.TestBroadReceive.action" />-->
</intent-filter>
</receiver>

动态注册

注册方式:在代码中调用​​Context.registerReceiver()​​方法

private  TestBroadReceive testBroadReceive;
// 选择在Activity生命周期方法中的onResume()中注册
@Override
protected void onResume(){
super.onResume();

// 1. 实例化BroadcastReceiver子类 & IntentFilter
testBroadReceive = new TestBroadReceive();
IntentFilter intentFilter = new IntentFilter();

// 2. 设置接收广播的类型
intentFilter.addAction("com.javayihao.demo.TestBroadReceive.action");

// 3. 动态注册:调用Context的registerReceiver()方法
registerReceiver(testBroadReceive, intentFilter);
}


// 注册广播后,要在相应位置记得销毁广播
// 即在onPause() 中unregisterReceiver(mBroadcastReceiver)
// 当此Activity实例化时,会动态将MyBroadcastReceiver注册到系统中
// 当此Activity销毁时,动态注册的MyBroadcastReceiver将不再接收到相应的广播。
@Override
protected void onPause() {
super.onPause();
//销毁在onResume()方法中的广播
unregisterReceiver(testBroadReceive);
}

当此 ​​App​​首次启动时,系统会自动实例化​​TestBroadcastReceiver​​类,并注册到系统中。

注意:

  • 动态广播最好在​​Activity​​​ 的​​onResume()​​​注册、​​onPause()​​注销。
  • 原因:

对于动态广播,有注册就必然得有注销,否则会导致内存泄露

 

Activity生命周期的方法是成对出现的:

  • onCreate() & onDestory()
  • onStart() & onStop()
  • onResume() & onPause()

在onResume()注册、onPause()注销是因为onPause()在App死亡前一定会被执行,从而保证广播在App死亡前一定会被注销,从而防止内存泄露。

  1. 不在onCreate() & onDestory() 或 onStart() & onStop()注册、注销是因为:
    当系统因为内存不足(优先级更高的应用需要内存,请看上图红框)要回收Activity占用的资源时,Activity在执行完onPause()方法后就会被销毁,有些生命周期方法onStop(),onDestory()就不会执行。当再回到此Activity时,是从onCreate方法开始执行。
  2. 假设我们将广播的注销放在onStop(),onDestory()方法里的话,有可能在Activity被销毁后还未执行onStop(),onDestory()方法,即广播仍还未注销,从而导致内存泄露。
  3. 但是,onPause()一定会被执行,从而保证了广播在App死亡前一定会被注销,从而防止内存泄露。

2.3.静态注册和动态注册区别

Android四大组件之BroadcastReceiver_sd卡

3.发送广播

上面我们已经将广播接受者定义好了,并且也注册了(注册方式选其一),下面我们发送广播

广播的类型

  • 普通广播(​​Normal Broadcast​​)
  • 系统广播(​​System Broadcast​​)
  • 有序广播(​​Ordered Broadcast​​)
  • 粘性广播(​​Sticky Broadcast​​)
  • App应用内广播(​​Local Broadcast​​)

3.1普通广播

开发者自身定义 ​​intent​​​的广播(最常用),若被注册了的广播接收者中注册时​​intentFilter​​​的​​action​​​与上述匹配,则会接收此广播(即进行回调​​onReceive()​​​)。如下​​TestBroadcastReceiver​​则会接收上述广播.若发送广播有相应权限,那么广播接收者也需要相应权限

public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setNormalBC();
}
/**
* 发送一个普通广播
*/
public void setNormalBC(){
Intent intent = new Intent("com.javayihao.demo.TestBroadReceive.action");
intent.putExtra("message","hello world!!!!!!!!!!!!!!!!!!!");
sendBroadcast(intent);
}

3.2有序广播

发送出去的广播被广播接收者按照先后顺序接收,有序是针对广播接收者而言的,

  • 广播接受者接收广播的顺序规则(同时面向静态和动态注册的广播接受者)
  1. 按照Priority属性值从大-小排序;
  2. Priority属性相同者,动态注册的广播优先;
  • 特点
  1. 接收广播按顺序接收
  2. 先接收的广播接收者可以对广播进行截断,即后接收的广播接收者不再接收到此广播;
  3. 先接收的广播接收者可以对广播进行修改,那么后接收的广播接收者将接收到被修改后的广播
  • 具体使用
    有序广播的使用过程与普通广播非常类似,差异仅在于广播的发送方式:

//发送
sendOrderedBroadcast(intent);
//终止
abortBroadcast()

注意

* 在注册广播中的<intent-filter>中使用android:priority属性,这个属性的范围在-1000---1000。
* 数值越大,优先权越高,在广播接收者中使用setResultExtras方法将一个Bundle对象设置为结果集对象,传递到下一个接受者哪里,这样优先级低的接受者可以用getResultExtras获取到最新的经过处理的信息集合
* 使用sendOrderedBroadcast方法有序广播的时候,需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为null,则表示接收者要接收此广播,需声明指定权限,这样做是从安全角度考虑的,例如系统的短信就是有序的广播的形式,一个应用可能是具有拦截垃圾短信的功能,当短信到来的时候它可以先接收到短信,必要时候终止广播的传递,这样的软件就必须声明接收短信的权限。

 * 同级别接收是先后是随机的,在到级别低的收到广播,如果先接收到的把广播拦截了,同级别以外的接收者是无法收到该广播的
 * 在这个方法发来的广播(代码注册方式中),收到广播先后次序为:注明优先级的、代码优先级的、没有优先级的,如果都没有优先级,代码注册接收到优先

 

3.3.系统广播

  • Android中内置了多个系统广播:只要涉及到手机的基本操作(如开机、网络状态变化、拍照等等),都会发出相应的广播
  • 每个广播都有特定的Intent - Filter(包括具体的action),Android系统广播action如下:

常用的几个

1. 开机启动广播

    <receiver
android:name=".BootCompleteReceiver">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>

2. 网络变化广播

    <receiver
android:name=".NetworkStateReceiver">
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>

因为在isNetworkAvailable方法中我们使用到了网络状态相关的API,所以需要声明相关的权限才行,下面就是对应的权限声明:

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

3. 电量变化广播

 

   <receiver
android:name=".BatteryChangedReceiver">
<intent-filter>
<action android:name="android.intent.action.BATTERY_CHANGED"/>
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>

4. SD卡状态广播

   

 <receiver
android:name="com.itheima.sdcradlistener.SDCardReceiver">
<intent-filter >
<action android:name="android.intent.action.MEDIA_MOUNTED"/>
<action android:name="android.intent.action.MEDIA_UNMOUNTED"/>
<action android:name="android.intent.action.MEDIA_REMOVED"/>
<data android:scheme="file"/>
</intent-filter>
</receiver>

5. 监听应用的安装、卸载、更新

 

   <receiver
android:name="com.itheima.app.AppReceiver">
<intent-filter >
<action android:name="android.intent.action.PACKAGE_ADDED"/>
<action android:name="android.intent.action.PACKAGE_REPLACED"/>
<action android:name="android.intent.action.PACKAGE_REMOVED"/>
<data android:scheme="package"/>
</intent-filter>
</receiver>

其他如下

常用系统广播的用法

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

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

Intent.ACTION_BATTERY_LOW;
//表示电池电量低

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

Intent.ACTION_MANAGE_PACKAGE_STORAGE;
//

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

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

Intent.ACTION_MEDIA_CHECKING;
//插入外部储存装置,比如SD卡时,系统会检验SD卡,此时发出的广播?
Intent.ACTION_MEDIA_EJECT;
//已拔掉外部大容量储存设备发出的广播(比如SD卡,或移动硬盘),不管有没有正确卸载都会发出此广播?
//广播:用户想要移除扩展介质(拔掉扩展卡)。
Intent.ACTION_MEDIA_MOUNTED;
//插入SD卡并且已正确安装(识别)时发出的广播
//广播:扩展介质被插入,而且已经被挂载。
Intent.ACTION_MEDIA_NOFS;
//
Intent.ACTION_MEDIA_REMOVED;
//外部储存设备已被移除,不管有没正确卸载,都会发出此广播?
// 广播:扩展介质被移除。
Intent.ACTION_MEDIA_SCANNER_FINISHED;
//广播:已经扫描完介质的一个目录
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;
//
Intent.ACTION_MEDIA_SCANNER_STARTED;
//广播:开始扫描介质的一个目录

Intent.ACTION_MEDIA_SHARED;
// 广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。
Intent.ACTION_MEDIA_UNMOUNTABLE;
//
Intent.ACTION_MEDIA_UNMOUNTED
// 广播:扩展介质存在,但是还没有被挂载 (mount)。
Intent.ACTION_NEW_OUTGOING_CALL;

Intent.ACTION_PACKAGE_ADDED;
//成功的安装APK之后
//广播:设备上新安装了一个应用程序包。
//一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)
Intent.ACTION_PACKAGE_CHANGED;
//一个已存在的应用程序包已经改变,包括包名
Intent.ACTION_PACKAGE_DATA_CLEARED;
//清除一个应用程序的数据时发出的广播(在设置--应用管理--选中某个应用,之后点清除数据时?)
//用户已经清除一个包的数据,包括包名(清除包程序不能接收到这个广播)

Intent.ACTION_PACKAGE_INSTALL;
//触发一个下载并且完成安装时发出的广播,比如在电子市场里下载应用?
//
Intent.ACTION_PACKAGE_REMOVED;
//成功的删除某个APK之后发出的广播
//一个已存在的应用程序包已经从设备上移除,包括包名(正在被安装的包程序不能接收到这个广播)

Intent.ACTION_PACKAGE_REPLACED;
//替换一个现有的安装包时发出的广播(不管现在安装的APP比之前的新还是旧,都会发出此广播?)
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;
//时间被设置时发出的广播

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

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

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

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

Intent.ACTION_USER_PRESENT;
//

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

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

  • 背景
    Android中的广播可以跨App直接通信(exported对于有intent-filter情况下默认值为true)
  • 冲突
    可能出现的问题:
  • 其他App针对性发出与当前App intent-filter相匹配的广播,由此导致当前App不断接收广播并处理;
  • 其他App注册与当前App一致的intent-filter用于接收广播,获取广播具体信息;
    即会出现安全性 & 效率性的问题。
  • 解决方案
    使用App应用内广播(Local Broadcast)
  1. App应用内广播可理解为一种局部广播,广播的发送者和接收者都同属于一个App。
  2. 相比于全局广播(普通广播),App应用内广播优势体现在:安全性高 & 效率高
  • 具体使用1 - 将全局广播设置成局部广播
  1. 注册广播时将exported属性设置为false,使得非本App内部发出的此广播不被接收;
  2. 在广播发送和接收时,增设相应权限permission,用于权限验证;
  3. 发送广播时指定该广播接收器所在的包名,此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中。

通过intent.setPackage(packageName)指定报名

  • 具体使用2 - 使用封装好的LocalBroadcastManager类
    使用方式上与全局广播几乎相同,只是注册/取消注册广播接收器和发送广播时将参数的context变成了LocalBroadcastManager的单一实例

注:对于LocalBroadcastManager方式发送的应用内广播,只能通过LocalBroadcastManager动态注册,不能静态注册

//注册应用内广播接收器
//步骤1:实例化BroadcastReceiver子类 & IntentFilter TestBroadcastReceiver
TestBroadcastReceiver = new TestBroadcastReceiver ();
IntentFilter intentFilter = new IntentFilter();

//步骤2:实例化LocalBroadcastManager的实例
localBroadcastManager = LocalBroadcastManager.getInstance(this);

//步骤3:设置接收广播的类型
intentFilter.addAction(android.net.conn.CONNECTIVITY_CHANGE);

//步骤4:调用LocalBroadcastManager单一实例的registerReceiver()方法进行动态注册
localBroadcastManager.registerReceiver(TestBroadcastReceiver , intentFilter);

//取消注册应用内广播接收器
localBroadcastManager.unregisterReceiver(TestBroadcastReceiver );

//发送应用内广播
Intent intent = new Intent();
intent.setAction(BROADCAST_ACTION);
localBroadcastManager.sendBroadcast(intent);

5. 粘性广播(Sticky Broadcast)

由于在Android5.0 & API 21中已经失效,所以不建议使用,在这里也不作过多的总结。

特别注意

对于不同注册方式的广播接收器回调OnReceive(Context context,Intent intent)中的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;

 

 

参考链接

​​https://www.jianshu.com/p/ca3d87a4cdf3​​

 

 

举报

相关推荐

0 条评论