Android的广播机制:BroadcastReceiver

Android中的广播主要可以分为两种类型:标准广播和有序广播

标准广播为异步,有序广播为同步

接收系统广播(动态监听时间变化):

我们可以根据自己感兴趣的广播,自由地注册BroadcastReceiver,这样当有相应的广播发出时,相应的BroadcastReceiver就能够收到该广播,并可以在内部进行逻辑处理。注册
BroadcastReceiver的方式一般有两种:在代码中注册和在AndroidManifest.xml中注册。其中前者也被称为动态注册,后者也被称为静态注册。

动态注册实现监听时间变化

创建一个BroadcastReceiver的方法:

其实只需新建一个类,让它继承自BroadcastReceiver,并重写父类的onReceive()方法就行了。这样当有广播到来时,onReceive()方法就会得到执行,具体的逻辑就可以在这个方法中处理。

class MainActivity : AppCompatActivity() {

    lateinit var timeChangeReceiver: TimeChangeReceiver

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        val intentFilter = IntentFilter()
        intentFilter.addAction("android.intent.action.TIME_TICK")
        timeChangeReceiver = TimeChangeReceiver()
        registerReceiver(timeChangeReceiver, intentFilter)
    }

    override fun onDestroy() {
        super.onDestroy()
        unregisterReceiver(timeChangeReceiver)
    }

    inner class TimeChangeReceiver : BroadcastReceiver() {
        override fun onReceive(context: Context?, intent: Intent?) {
            val timeText: TextView = findViewById(R.id.timeText)
            var timeNum = timeText.text.toString().toInt()
            timeText.text = (timeNum+1).toString()
            Toast.makeText(context,"Time has changed to $timeNum",Toast.LENGTH_SHORT).show()
        }

    }
}

在MainActivity中定义了一个内部类TimeChangeReceiver,这个类是继承自BroadcastReceiver的,并重写了父类的onReceive()方法。这样每当系统时间发生变化时,onReceive()方法就会得到执行。

onCreate()方法中,首先我们创建了一个IntentFilter的实例,并给它添加了一个值为android.intent.action.TIME_TICK的action,这是系统在时间变化时的系统广播。也就是说我们的BroadcastReceiver想要监听什么广播,就在这里添加相应的action

接下来创建了一个TimeChangeReceiver的实例,然后调用registerReceiver()方法进行注册,将TimeChangeReceiver的实例和IntentFilter的实例都传了进去,这样TimeChangeReceiver就会收到所有值为android.intent.action.TIME_TICK的广播,也就实现了监听系统时间变化的功能。

最后要记得,动态注册的BroadcastReceiver一定要取消注册才行!!!!这里我们是在onDestroy()方法中通过调用unregisterReceiver()方法来实现的。

静态注册实现开机启动:

动态注册的BroadcastReceiver可以自由地控制注册与注销,在灵活性方面有很大的优势。但是它存在着一个缺点,即必须在程序启动之后才能接收广播,因为注册的逻辑是写在onCreate()方法中的。

当我们想让程序在未启动时也能接受广播,就需要使用静态注册的方式。

其实从理论上来说,动态注册能监听到的系统广播,静态注册也应该能监听到,在过去的Android系统中确实是这样的。但是由于大量恶意的应用程序利用这个机制在程序未启动的情况下监听系统广播,从而使任何应用都可以频繁地从后台被唤醒,严重影响了用户手机的电量和性能,因此Android系统几乎每个版本都在削减静态注册BroadcastReceiver的功能。

在Android 8.0系统之后,所有隐式广播都不允许使用静态注册的方式来接收了。隐式广播指的是那些没有具体指定发送给哪个应用程序的广播,大多数系统广播属于隐式广播,但是少数特殊的系统广播目前仍然允许使用静态注册的方式来接收。这些特殊的系统广播列表详见

隐式广播例外情况  |  Android 开发者  |  Android Developers

android.intent.action.BOOT_COMPLETED 属于特殊的系统广播,为开机广播。

通过右击com.example.broadcasttest包→New→Other→Broadcast Receiver,Exported属性表示是否允许这个BroadcastReceiver接收本程序以外的广播,Enabled属性表示是否启用这个BroadcastReceiver。

静态的BroadcastReceiver一定要在AndroidManifest.xml文件中注册才可以使用。不过,由于我们是使用Android Studio的快捷方式创建的BroadcastReceiver,因此注册这一步已经自动完成了。

不过目前的BootCompleteReceiver是无法收到开机广播的,因为我们还需要对AndroidManifest.xml文件进行修改才行!!!

由于Android系统启动完成后会发出一条值为android.intent.action.BOOT_COMPLETED的广播,因此我们在<receiver>标签中又添加了一个<intent-filter>标签,并在里面声明了相应的action。

这里有非常重要的一点需要说明。Android 系统为了保护用户设备的安全和隐私,做了严格的规定:如果程序需要进行一些对用户来说比较敏感的操作,必须在AndroidManifest.xml文件中进行权限声明,否则程序将会直接崩溃。比如这里接收系统的开机广播就是需要进行权限声明的,所以我们在上述代码中使用<uses-permission>标签声明了android.permission.RECEIVE_BOOT_COMPLETED权限。

发送自定义广播:

1、发送标准广播:

创建一个 MyBroadcastReceiver ,并在onReceive()中加入如下代码:

    override fun onReceive(context: Context, intent: Intent) {
        Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).show()
    }

同时在  AndroidManifest.xml  中对这个 BroadcastReceiver 进行修改。如下:

        <receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="com.example.boardcastreceivertest.MY_BROADCAST" />
            </intent-filter>
        </receiver>

让这个BroadcastReceiver能够接收一条值为com.example.boardcastreceivertest.MY_BROADCAST的广播!!

在MainActivity中修改:

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        val button: Button = findViewById(R.id.button)
        button.setOnClickListener {
            val intent = Intent("com.example.boardcastreceivertest.MY_BROADCAST")
            intent.setPackage(packageName)
            sendBroadcast(intent)
        }
    }

首先新建一个Intent对象,并把要发送的广播值传入,然后调用setPackage() 方法,传入当前应用程序包名。最后调用sendBroadcast()方法将广播发送出去,这样所有监听com.example.broadcasttest.MY_BROADCAST这条广播的BroadcastReceiver就会收
到消息了。此时发出去的广播就是一条标准广播。

调用setPackage()的原因:在Android8.0系统之后,静态注册的BroadcastReceiver是无法接收隐式广播的,而默认情况下我们发出的自定义广播恰恰都是隐式广播。因此这里一定要调用setPackage()方法,指定这条广播是发送给哪个应用程序的,从而让它变成一条显式广播,否则静态注册的BroadcastReceiver将无法接收到这条广播。

2、发送有序广播:

创建一个新的BroadcastReceiver,新建AnotherBroadcastReceiver。

class AnotherBroadcastReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        Toast.makeText(context,"received in AnotherBroadcastReceiver",Toast.LENGTH_SHORT).show()
    }
}

然后在AndroidManifest.xml中对这个BroadcastReceiver的配置进行修改:

        <receiver
            android:name=".AnotherBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="100">
                <action android:name="com.example.boardcastreceivertest.MY_BROADCAST" />
            </intent-filter>
        </receiver>

AnotherBroadcastReceiver同样接收的是com.example.broadcasttest.MY_BROADCAST这条广播。

然后修改MainActivity中代码:

        button.setOnClickListener {
            val intent = Intent("com.example.boardcastreceivertest.MY_BROADCAST")
            intent.setPackage(packageName)
            sendOrderedBroadcast(intent,null)
        }

将sendBroadcast()修改为sendOrderedBroadcast().

sendOrderedBroadcast()方法接收两个参数:

第一个参数仍然是Intent;

第二个参数是一个与权限相关的字符串,这里传入null就行了。

有序广播可以进行截断,能够选择是否将广播继续传播那么如何实现先后顺序呢?

通过在AndroidManifest.xml中直接设置优先级,优先级高的先收到广播,设置方法:

        <receiver
            android:name=".AnotherBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="100">
                <action android:name="com.example.boardcastreceivertest.MY_BROADCAST" />
            </intent-filter>
        </receiver>

android:priority属性为设置优先级。

如何截断传播呢?

在高优先级的类中的onReceive()方法中调用了abortBroadcast()方法,就表示将这条广播截断,后面的BroadcastReceiver将无法再接收到这条广播!!!!!!

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值