# BroadcastReceiver 广播接收者必会必知
BroadcastReceiver 广播接收者 Android 四大组件之一,是 Android 系统提供的一种通讯方式。
我们举个形象的例子来帮我理解下 BroadcastReceiver,记得以前读书 的时候,每个班级都会有一个挂在墙上的大喇叭,用来广播一些通知,比如,开学要去搬书,教导主任对着大喇叭喊广播: "每个班级找几个同学教务处拿书",发出这个广播后,所有同学都会在同一时刻收到这条广播通知, 收到,但不是每个同学都会去搬书,一般去搬书的都是班里的 "大力士",这群 "大力士" 接到这条 广播后就会动身去把书搬回可是! —— 好吧,上面这个就是一个广播传递的一个很形象的例子: 教导主任喊大喇叭 --> 发送广播 --> 所有学生都能收到广播 --> 大力士处理广播 。这个流程涉及到两个角色,一个是广播发送者,一个是广播接收者。
回到 Android 中, 系统自己在很多时候都会发送广播,比如电量变化,wifi 连接变化,插入耳机,输入法改变等,系统都会发送广播,这个叫系统广播。此时系统就是广播发送者
如果我们的 APP 想要收到这些广播,这个时候我们只需要注册一个 BroadcastReceiver,当 wifi 连接发生变化,我们注册的广播就会收到通知~。此时我们的 APP 就是广播接收者
当然我们也可以自己发广播,比如:登录成功后发出广播,监听这个广播的接收者就可以做些刷新页面的动作。此时我们的 APP 既是广播发送者,也是广播接收者。
应用场景:
Android
不同组件间的通信(含 :应用内 / 不同应用之间)多线程通信
与
Android
系统在特定情况下的通信
# 两种广播类型
标准广播:发出广播后,该广播事件的接收者,几乎会在同一时刻收到通知,都可以响应或不响应该事件
有序广播:发出广播后,同一时刻,只有一个广播接收者能收到、一个接收者处理完后之后,可以选择继续向下传递给其它接收者,也可以拦截掉广播。[不常用、不推荐使用了]
# 监听系统网络连接变化
# 定义一个广播接收者
// 监听网络连接状态的变化,并 toast 提示 | |
class TestBroadcastReceiver : BroadcastReceiver() { | |
override fun onReceive(context: Context?, intent: Intent?) { | |
if (intent?.action?.equals(ConnectivityManager.CONNECTIVITY_ACTION)==true){ | |
val connectivityManager:ConnectivityManager = context?.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager | |
val info = connectivityManager.activeNetworkInfo | |
if(info != null && info.isAvailable) { | |
val typeName = info.typeName | |
Toast.makeText(context,"当前网络名称:${typeName}",Toast.LENGTH_LONG).show() | |
} else { | |
Toast.makeText(context,"当前无网络连接",Toast.LENGTH_LONG).show() | |
} | |
} | |
} | |
} |
# 运行时动态注册广播接收事件
class TestBroadcastRecevierActivity :AppCompatActivity(){ | |
private lateinit var myReceiver: TestBroadcastReceiver | |
override fun onCreate(savedInstanceState: Bundle?) { | |
super.onCreate(savedInstanceState) | |
myReceiver = TestBroadcastReceiver() | |
// 创建广播过滤器,指定只接收 android.net.conn.CONNECTIVITY_CHANGE 的广播事件 | |
val intentFilter = IntentFilter() | |
intentFilter.addAction(Intent.ACTION_CONFIGURATION_CHANGED) | |
registerReceiver(myReceiver, intentFilter) | |
} | |
override fun onDestroy() { | |
super.onDestroy() | |
unregisterReceiver(myReceiver) // 必须要在 onDestroy 时反注册,否则会内存泄漏 | |
} | |
} |
note
不要在收到广播后进行任何耗时操作,因为在广播中是不允许开辟线程的,当 onReceiver () 方法运行较长时间 (超过 10 秒) 还没有结束的话,那么程序会报错 (ANR), 广播更多的时候扮演的是一个打开其他组件的角色,比如启动 Service,Notification 提示,Activity 等!
# 静态注册广播
AndroidManifest.xml 中注册广播
<receiver android:name=".components.TestBroadcastReceiver"> | |
<intent-filter > | |
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/> | |
</intent-filter> | |
</receiver> |
Google 官方声明:Beginning with Android 8.0 (API level 26), the system imposes additional restrictions on manifest-declared receivers. If your app targets API level 26 or higher, you cannot use the manifest to declare a receiver for most implicit broadcasts (broadcasts that do not target your app specifically). 大概意思就是说:从 android 8.0(API26)开始,对清单文件中静态注册广播接收者增加了限制,建议大家不要在清单文件中静态注册广播接收者。
** 其实说白点:** 就是因为在清单文件中静态注册广播接收者,容易让一些 "不法分子" 获取用户的隐私 (如:电话监听、短信监听等等),所以 google 限制了静态注册(Android 在保护用户隐私上坚持不懈的努力着... 也许 google 还要其他的考虑吧。咱也不知道... 咱也不敢问😁)
# 解决静态注册广播接收者收不到事件的问题
虽然从 Android8.0 开始,系统明确限制了静态广播注册去监听系统行为,但是应用自己定义的的广播事件还是可以使用静态注册的
val intent= Intent(); | |
intent.action = "com.example.firstapp.TEST_BROADCAST_RECEVIER"; | |
// 下面这一行在 Android 7.0 及以下版本不是必须的,但是 Android 8.0 或者更高版本,发送广播的条件更加严苛,必须添加这一行内容。 | |
// 创建的 ComponentName 实例化对象有两个参数,第 1 个参数是指接收广播类的包名,第 2 个参数是指接收广播类的完整类名。 | |
intent.component =ComponentName(packageName,"com.example.firstapp.component.TEST_BROADCAST_RECEVIER") | |
sendBroadcast(intent) |
# 发送自定义事件广播
上面我们都是接收系统的广播,系统发,我们收,我们不能老这么被动,总得主动点是吧!
# 全局发送广播
全局发送广播,如果别人家 App 也注册了该事件监听,也能收到,比较不合理。
sendBroadcast(new Intent("com.example.firstapp.component.TEST_BROADCAST_RECEVIER")); |
# 应用内发送广播
- App 应用内广播可理解为一种局部广播,广播的发送者和接收者都同属于一个 App。
- 相比于全局广播(普通广播),App 应用内广播优势体现在:安全性高 & 效率高
// 使用 LocalBroadcastManager 来注册应用内广播 | |
LocalBroadcastManager.getInstance(this).registerReceiver(myReceiver, itFilter) | |
// 使用 LocalBroadcastManager 来发送应用内广播 | |
LocalBroadcastManager.getInstance(this).sendBroadcast(intent) |
# 系统广播(System Broadcast)
- Android 中内置了多个系统广播:只要涉及到手机的基本操作(如开机、网络状态变化、拍照等等),都会发出相应的广播
- 每个广播都有特定的 Intent - Filter(包括具体的 action),Android 系统广播 action 如下:
系统操作 | action |
---|---|
监听网络变化 | android.net.conn.CONNECTIVITY_CHANGE |
关闭或打开飞行模式 | Intent.ACTION_AIRPLANE_MODE_CHANGED |
充电时或电量发生变化 | Intent.ACTION_BATTERY_CHANGED |
电池电量低 | Intent.ACTION_BATTERY_LOW |
电池电量充足(即从电量低变化到饱满时会发出广播 | Intent.ACTION_BATTERY_OKAY |
系统启动完成后 (仅广播一次) | Intent.ACTION_BOOT_COMPLETED |
按下照相时的拍照按键 (硬件按键) 时 | Intent.ACTION_CAMERA_BUTTON |
屏幕锁屏 | Intent.ACTION_CLOSE_SYSTEM_DIALOGS |
设备当前设置被改变时 (界面语言、设备方向等) | Intent.ACTION_CONFIGURATION_CHANGED |
插入耳机时 | Intent.ACTION_HEADSET_PLUG |
未正确移除 SD 卡但已取出来时 (正确移除方法:设置 --SD 卡和设备内存 -- 卸载 SD 卡) | Intent.ACTION_MEDIA_BAD_REMOVAL |
插入外部储存装置(如 SD 卡) | Intent.ACTION_MEDIA_CHECKING |
成功安装 APK | Intent.ACTION_PACKAGE_ADDED |
成功删除 APK | Intent.ACTION_PACKAGE_REMOVED |
重启设备 | Intent.ACTION_REBOOT |
屏幕被关闭 | Intent.ACTION_SCREEN_OFF |
屏幕被打开 | Intent.ACTION_SCREEN_ON |
关闭系统时 | Intent.ACTION_SHUTDOWN |
重启设备 | Intent.ACTION_REBOOT |