Android--Broadcast使用

本篇主要总结了广播的使用,包括常用系统广播列举、动态注册广播、静态注册广播和自定义发送广播的示例。

常用系统广播

系统广播常量的声明和定义在Intent.java中。

  1. 系统启动完成
ACTION_BOOT_COMPLETED = "android.intent.action.BOOT_COMPLETED"
  1. 重启设备
ACTION_REBOOT = "android.intent.action.REBOOT"
  1. 息屏
ACTION_SCREEN_OFF = "android.intent.action.SCREEN_OFF"
  1. 亮屏
ACTION_SCREEN_ON = "android.intent.action.SCREEN_ON"
  1. 屏幕锁屏
ACTION_CLOSE_SYSTEM_DIALOGS = "android.intent.action.CLOSE_SYSTEM_DIALOGS"
  1. 日期改变
ACTION_DATE_CHANGED = "android.intent.action.DATE_CHANGED"
  1. 时间改变(1min)
ACTION_TIME_TICK = "android.intent.action.TIME_TICK"
  • 注:系统启动广播可以用来实现自启动apk或服务等功能,自启动apk也可以在AndroidManifest.xml中添加home属性实现。
  <intent-filter>
    <action android:name="android.intent.action.MAIN" />
    <category android:name="android.intent.category.HOME" />
    <category android:name="android.intent.category.DEFAULT" />
  </intent-filter>

动态注册广播

以时间变化的系统广播为例,动态注册广播可分为3步:

  1. 创建广播接收器BroadcastReceiver;
  2. 创建Intent过滤器实例,给IntentFilter实例添加系统广播的action;
  3. 在使用的Activity中动态注册和取消注册;
  • 创建BroadcastReceiver

创建一个类继承自BroadcastReceiver,重写父类的onReceive()方法,当接收到广播时该方法就会执行,可以通过intent的action区分不同的广播,执行对应操作。

<MyBroadcastReceiver.kt>

class MyBroadcastReceiver : BroadcastReceiver(){

    override fun onReceive(context: Context?, intent: Intent?) {
        // 接收到广播就会执行onReceive
        if (intent == null) {
            return
        }
        
        Log.d("=========", "onReceive: ${intent.action} ========")
        if (intent.action == Intent.ACTION_TIME_TICK){
            Toast.makeText(context, "time has changed", Toast.LENGTH_SHORT).show()
        }
    }

}
  • 创建IntentFilter实例

创建IntentFilter实例,给该实例添加需要监听的系统广播的action,这样当系统时间改变发出这条广播时,当前组件就可以在接收器的onReceive()中接收到。

<MainActivity.kt>

class MainActivity : AppCompatActivity() {

    private lateinit var myBroadcastReceiver: MyBroadcastReceiver

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

        initBroadcastReceiver()
    }

    private fun initBroadcastReceiver(){
        // 创建IntentFilter实例
        val intentFilter = IntentFilter()
        intentFilter.addAction("android.intent.action.TIME_TICK")
        myBroadcastReceiver = MyBroadcastReceiver()
        // 动态注册系统广播
        registerReceiver(myBroadcastReceiver, intentFilter)
    }

    override fun onDestroy() {
        super.onDestroy()
        // 取消广播注册
        unregisterReceiver(myBroadcastReceiver)
    }
}
  • 动态注册和取消注册

动态注册广播时,需要将接收器实例和intent过滤器实例都传入到registerReceiver()中,动态注册的广播接收器系统不会回收,使用完成后需要在onDestroy()中unregisterReceiver()取消注册.

静态注册广播

动态注册必须在应用程序启动后才能接收到广播,有些操作需要在设备一开机就执行,这种情况就需要使用静态注册的方式,以开机广播为例,静态注册广播也分为3步:

  1. 创建广播接收器BroadcastReceiver;
  2. 在AndroidManifest.xml中注册静态广播;
  3. 添加广播权限;
  • 创建BroadcastReceiver

注册静态广播可借助AS工具来创建BroadcastReceiver类,右击包文件夹--New--Other--BroadcastReceiver,在弹窗中修改类名确认即可,这种方法创建的BroadcastReceiver类会自动注册在AndroidManifest.xml中。

4.png
5.png

然后重写父类的onReceive()方法处理接收到的广播消息,注意,由于onReceive()时运行在主线程中,需要避免执行耗时操作。

<MyBroadcastReceiver.kt>

class MyBroadcastReceiver : BroadcastReceiver(){

    override fun onReceive(context: Context?, intent: Intent?) {
        // 接收到广播就会执行onReceive
        if (intent == null) {
            return
        }

        Log.d("=========", "onReceive: ${intent.action} ========")
        if (intent.action == Intent.ACTION_BOOT_COMPLETED){
            Toast.makeText(context, "system started", Toast.LENGTH_SHORT).show()
        }
    }

}
  • 在AndroidManifest.xml中注册静态广播

与动态注册类似,需要在广播注册时添加系统广播的action。

<AndroidManifest.xml>

    <!-- 静态注册系统广播 -->
    <receiver android:name=".MyBroadcastReceiver"
        android:enabled="true"
        android:exported="true">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED"/>
        </intent-filter>
    </receiver>
  • 添加广播权限

大多数系统广播属于隐式广播,不指定时发送给哪个应用程序的广播,为了防止不良应用程序通过系统广播主动从后台唤醒,Android 8.0 之后静态注册隐式广播需要添加广播权限。

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

自定义广播

广播有标准广播和有序广播两种类型:

  1. 标准广播:是一种异步执行广播,广播发出后所有BroadcastReceiver几乎可以同时收到,广播接收器接收广播无先后顺序;

  2. 有序广播:是一种同步执行广播,广播发出后同一时刻只有一个BroadcastReceiver能接收到,这个广播接收机逻辑执行完毕之后才会传给下一个,接收器接收的先后顺序可以通过优先级来设置,并且可以截断广播,不继续往后发。

  • 发送标准广播
  1. 现在已经有一个MyBroadcastReceiver广播接收器了,再创建一个SecondBroadcastReceiver用于测试标准广播,所有接收器都可以同时接收到广播,且无先后顺序的特征。

<SecondBroadcastReceiver.kt>

class SecondBroadcastReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        val data = intent.getStringExtra("data")
        Log.d("=========", "SecondBroadcastReceiver: ${intent.action} ,data = $data ========")
    }

}
  1. 静态注册广播时添加action,action内容可以自行定义,只要接收和发送使用的action统一即可。

<AndroidManifest.xml>

        <!-- 静态注册系统广播 -->
        <receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="com.example.broadcastDemo_test" />
            </intent-filter>
        </receiver>

        <receiver
            android:name=".SecondBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="com.example.broadcastDemo_test" />
            </intent-filter>        
        </receiver>
  1. 页面中添加一个按钮,通过点击按钮来发送广播:

<MainActivity.kt>

    private fun initView(){
        btn.setOnClickListener{
            val intent = Intent("com.example.broadcastDemo_test")
            // 广播可携带数据
            intent.putExtra("data", "Broadcast Demo Test!")
            intent.setPackage(packageName)
            // 发送标准广播
            sendBroadcast(intent)
        }
    }

运行之后点击按钮发送广播,两个接收器在同一时间接收到了自定义发送的标准广播。

6.png
  • 发送有序广播

有序广播只需要在标准广播的基础上稍作修改即可实现。

  1. 发送标准广播使用的是sendBroadcast()方法,发送有序广播需要改为sendOrderedBroadcast()方法。

<MainActivity.kt>

    private fun initView(){
        btn.setOnClickListener{
            val intent = Intent("com.example.broadcastDemo_test")
            // 广播可携带数据
            intent.putExtra("data", "Broadcast Demo Test!")
            intent.setPackage(packageName)
            // 有序广播
            sendOrderedBroadcast(intent, null)
        }
    }
  1. 有序广播是有先后顺序的,可以在intent的filter中设置接收器的优先级,这里将SecondBroadcastReceiver的优先级设置为100。

<AndroidManifest.xml>

        <receiver
            android:name=".SecondBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="100">
                <action android:name="com.example.broadcastDemo_test" />
            </intent-filter>
        </receiver>
  1. SecondBroadcastReceiver优先接收到广播,并且有权决定广播之后发送给其他接收器,如果不希望广播继续发送的话可以截断广播。

<SecondBroadcastReceiver.kt>

class SecondBroadcastReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {

        val data = intent.getStringExtra("data")
        Log.d("=========", "SecondBroadcastReceiver: ${intent.action}, data = $data ========")

        // 广播截断
        abortBroadcast()
    }
}

运行之后就只有优先级高的SecondBroadcastReceiver的打印,广播被截断了,MyBroadcastReceiver根本没有收到广播,因此不会打印内容。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容