# 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"));

# 应用内发送广播

  1. App 应用内广播可理解为一种局部广播,广播的发送者和接收者都同属于一个 App。
  2. 相比于全局广播(普通广播),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
成功安装 APKIntent.ACTION_PACKAGE_ADDED
成功删除 APKIntent.ACTION_PACKAGE_REMOVED
重启设备Intent.ACTION_REBOOT
屏幕被关闭Intent.ACTION_SCREEN_OFF
屏幕被打开Intent.ACTION_SCREEN_ON
关闭系统时Intent.ACTION_SHUTDOWN
重启设备Intent.ACTION_REBOOT