Android 广播机制详解

600 阅读6分钟

我们知道,Android 的四大组件为 Activity,Service,BroadcastReceiver 和 ContentProvider,广播作为四大组件之一,应用场景相当多。Android 中每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就会收到自己所关心的广播内容,这些广播可以来自系统,也可以来自其他应用程序。

接收系统广播

Android 内置了多个系统广播,每个系统广播都具有特定的 intent-filter,有具体的 action,系统广播发出后,被相应的 BroadcastReceiver 接收,系统广播在系统内部当特定事件发生时,由系统自动发出。

动态注册

动态注册就是在代码中注册,这样能够灵活地实现广播的注册和注销,但是必须要程序启动后才能接收到。动态注册的广播不会自动销毁,所以不用的时候别忘了手动销毁。

这里以监听网络变化为例

class NetChangeReceiver : BroadcastReceiver() {
    override fun onReceive(p0: Context, p1: Intent) {
        Log.i(tag, "network state change")
    }
}
class MainActivity : AppCompatActivity() {

    private val netChangeReceiver = NetChangeReceiver()

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        //注册
        val intentFilter = IntentFilter("android.net.conn.CONNECTIVITY_CHANGE")
        registerReceiver(netChangeReceiver, intentFilter)
    }

    override fun onDestroy() {
        super.onDestroy()
        //取消注册
        unregisterReceiver(netChangeReceiver)
    }
}

静态注册

静态注册是在 AndroidManifest 中注册的,不需要手动取消注册,可以在程序还没启动的时候接收到注册的广播。

这里以接收开机广播为例,在手机开机的时候,我们的应用程序肯定是没有启动的,这时就应该用静态注册来接收系统的开机广播。

首先使用 Android Studio 创建广播接收器

1.png

创建之后,就会生成如下代码,这里弹个 Toast

class MyReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        Toast.makeText(context, "Start Up", Toast.LENGTH_SHORT).show()
    }
}
<receiver
    android:name=".MyReceiver"
    android:enabled="true"
    android:exported="true"></receiver>

不过这还不够,我们需要在 AndroidManifest 中添加权限和广播的 intent-filter

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<receiver
    android:name=".MyReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter>
        <action android:name="android.intent.action.BOOT_COMPLETED" />
    </intent-filter>
</receiver>

enabled 表示是否可以被系统实例化,设置为 true 才会被激活。exported 表示当前 BroadcastReceiver 是否可以从应用外部获取信息。也就是说,如果为 false,当前 BroadcastReceive 只能收到同一个应用发出的广播。

发送自定义广播

Android 中广播分为两种类型:标准广播和有序广播。

  • 标准广播:一种完全异步执行的广播,在广播发出之后,所有的 BroadcastReceiver 几乎会在同一时刻收到这条广播消息,没有任何先后顺序可言,并且无法被截断的。
  • 有序广播:一种同步执行的广播,在广播发出之后,同一时刻只会有一个 BroadcastReceiver 收到这条广播消息,当这个 BroadcastReceiver 中的逻辑执行完毕后,广播才会继续传递。所以此时的 BroadcastReceiver 是有先后顺序的,优先级高的 BroadcastReceiver 先收到广播消息,并且前面的 BroadcastReceiver 还可以截断正在传递的广播,这样后面的 BroadcastReceiver 就无法收到广播消息了。

我们先来发送一个标准广播,先定义一个广播接收器,这里接收后就弹个 Toast

class CustomReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        Toast.makeText(context, "receive a broadcast", Toast.LENGTH_SHORT).show()
    }
}

这里让它接收一个值为 com.android.jian 的广播

<receiver
    android:name=".CustomReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter>
        <action android:name="com.android.jian" />
    </intent-filter>
</receiver>

发送广播时一定要设置它的 package,在 Android 8 以后,静态注册的 BroadcastReceiver 是无法接收隐式广播的,而默认情况下我们发出的自定义广播都是隐式广播。因此这里要设置它的 package ,指定这条广播是发送给哪个应用程序的,从而让它变成一条显式广播。但是,动态注册的广播又无需指定 package 就可以接收到。

val intent = Intent("com.android.jian")
//这个是必须的,如果想发送到别的应用,就配置别的应用的 packageName 即可。
intent.`package` = packageName
sendBroadcast(intent)

接下来,我们来发送一个有序广播,上面提到,有序广播是可以被截断的,所以这里再创建一个 BroadcastReceiver

class OrderedReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        Toast.makeText(context, "receive a Ordered broadcast", Toast.LENGTH_SHORT).show()
    }
}

这里设置广播的优先级,保证它优先收到广播。

<receiver
    android:name=".OrderedReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter android:priority="100">
        <action android:name="com.android.jian" />
    </intent-filter>
</receiver>

但是,现在这样的话,CustomReceiver 和 OrderedReceiver 都会收到这条广播,如果只想要 OrderedReceiver 收到的话,可以截断。

class OrderedReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        Toast.makeText(context, "receive a Ordered broadcast", Toast.LENGTH_SHORT).show()
        //在此截断广播
        abortBroadcast()
    }
}

这样,发送的广播就只有 OrderedReceiver 能收到了,因为它的优先级比较高,而且接收后截断了广播。

本地广播

使用该机制发出的广播只能在应用程序内部进行传递,并且广播接收器也只能接收来自本地应用程序发出的广播。

class MainActivity : AppCompatActivity() {
    
    private lateinit var localBroadcastManager: LocalBroadcastManager
    private val localReceiver = LocalReceiver()

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        localBroadcastManager = LocalBroadcastManager.getInstance(this)
        //注册本地广播接收器
        val intentFilter = IntentFilter("com.local.broadcast")
        localBroadcastManager.registerReceiver(localReceiver, intentFilter)

        findViewById<Button>(R.id.btn).setOnClickListener {
            //发送本地广播
            localBroadcastManager.sendBroadcast(Intent("com.local.broadcast"))
        }
    }

    override fun onDestroy() {
        super.onDestroy()
        //取消注册
        localBroadcastManager.unregisterReceiver(localReceiver)
    }

    class LocalReceiver : BroadcastReceiver() {
        override fun onReceive(p0: Context?, p1: Intent?) {
            Toast.makeText(p0, "LocalReceiver", Toast.LENGTH_SHORT).show()
        }
    }

}

如果广播不用跨应用的话,个人建议都使用本地广播好,可以避免一些安全漏洞,比如我们的应用发送的 Intent 只是简单地通过 action 进行匹配的话,恶意应用便可以注册一个广播接收者嗅探拦截到这个广播,如果这个广播里存在敏感数据,就被恶意应用窃取了。再比如,暴露的 BroadcastReceiver 对外接收 Intent 的话,如果恶意应用构造恶意的消息放在 Intent 中传输,被我们的 BroadcastReceiver 接收就有可能产生安全隐患。此外,也可以使用自定义广播权限来增加广播的安全性。

自定义广播权限

我们可以在 AndroidManifest 中自定义权限

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

<permission
    android:name="com.custom.permission"
    android:label="BroadcastReceiverPermission"
    android:protectionLevel="signature" />

protectionLevel 属性常用的如下:

  • normal:当某个 App 申请这个权限时系统会自动授予 App 权限,而不用询问用户。
  • dangerous:用于高风险的权限,例如访问短信,联系人,或使用摄像头等等,当某个 App 申请使用这个权限时,system 可能会询问用户是否同意。
  • signature:如果申请权限的 App 和声明权限的 App 有相同的证书,那么系统就会自动授予这个 App 权限,不用询问用户。

然后配置到广播接收器

<receiver
    android:name=".CustomReceiver"
    android:enabled="true"
    android:exported="true"
    android:permission="com.custom.permission">
    <intent-filter>
        <action android:name="com.android.jian" />
    </intent-filter>
</receiver>

如果是动态注册的话,这样

val customReceiver = CustomReceiver()
val intentFilter = IntentFilter("com.android.jian")
registerReceiver(customReceiver, intentFilter, "com.custom.permission", null)

发送广播的时候需要携带这个权限

val intent = Intent("com.android.jian")
sendBroadcast(intent, "com.custom.permission")

BroadcastReceiver 中的 Context

BroadcastReceiver 本身不是 Context,但在 onReceive 中有 context 参数。这个 context 随着注册方式的不同而不同。静态注册的 context 为 ReceiverRestrictedContext,动态注册的 context 为 Activity 的 context,LocalBroadcastManager 动态注册的 context 为 Application 的 context。