Android-Kotlin-Broadcast技术点

Android中为了便于系统级别的消息通知,引入了广播机制。如果想接收到广播就必须要注册广播接收者-BroadcastReceiver

1.广播的分类

  • 标准广播:是一种完全异步的广播,发出后所有的接收者会几乎同时接收到,没有向后顺序可言,这种广播效率比较高。
  • 有序广播:是一种同步广播,同一时刻只有一个接收者接收到这个广播,并且有优先级的概念,优先级高的接收者先接收到这个广播。接收到这个广播的接收者可以终止这个广播。

2.注册广播

  • 动态注册:在代码中注册的接收者,需要在相对应的声明周期里边进行反注册。

      override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
            setContentView(R.layout.activity_broadcast)
            register()
        }
    
    /**
     * 动态注册广播
     */
    private fun register() {
        val intentFilter=IntentFilter()
        intentFilter.addAction("android.intent.action.TIME_TICK")
        receiver=TimeChangeReceiver()
        registerReceiver(receiver,intentFilter)
    }
    
    
    inner class TimeChangeReceiver :BroadcastReceiver(){
        override fun onReceive(context: Context?, intent: Intent?) {
             showToast("time change")
        }
    
    }
    
      override fun onDestroy() {
            super.onDestroy()
            //取消注册广播
            unregisterReceiver(receiver)
        }
    
    
  • 静态注册:在AndroidManifest清单文件注册的广播接收者,应用未启动也能收到广播,并且在广播中执行相关的代码逻辑。

    class BootCompleteReceiver : BroadcastReceiver() {
    
        override fun onReceive(context: Context, intent: Intent) {
            // This method is called when the BroadcastReceiver is receiving an Intent broadcast.
            showToast("reboot already")
        }
    }
    
      <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
      <application
                ...
                android:theme="@style/AppTheme">
    
            <receiver
                    android:name=".broadcast.BootCompleteReceiver"
                    android:enabled="true"
                    android:exported="true">
           <intent-filter>
                    <action android:name="android.intent.action.BOOT_COMPLETED"/>
                </intent-filter>
            </receiver>
    
         ...
        </application>
    
    

这样就注册了一个静态的广播,能接听到开机的广播并弹出Tooast。exported=“true” 表示允许接收除了本程序以外的广播,enabled=“true”表示开启这个广播接收者。

Android系统为了保护用户设备的安全和隐私:如果程序需要进行一些敏感的操作,必须声明权限,上面这个启动广播就需要声明权限否则会崩溃。

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

重启手机,便收到了开机广播。

3.隐式广播

隐式广播是指那些没有具体指定发送给那个应用程序的广播。Android系统8.0之后所有的隐式广播都不允许使用静态注册的方式注册来接收了。少数特殊的系统广播目前还能使用,比如:RECEIVE_BOOT_COMPLETED。我们发送的自定义广播也是隐式广播,如果希望能被自己的应用程序接收到,需要显式指定包名才可以。

4.发送广播

静态注册自定义的广播接收者,并指定Action

class MyReceiver : BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        // This method is called when the BroadcastReceiver is receiving an Intent broadcast.
       showToast("receive my Broadcast" )
    }
}

 //AndroidManifest添加
     <receiver
                android:name=".broadcast.MyReceiver"
                android:enabled="true"
                android:exported="true">
            <intent-filter>
                <action android:name="com.test.kotlin_test.MY_RECEIVER"/>
            </intent-filter>
        </receiver>

发送标准广播,注意需要设置应用的包名,intent.setPackage("com.test.kotlin_test")指明这个广播是发送给那个应用程序的,这样这个广播就不是隐式广播了,否则静态注册接收不到。

fun sendBroadcast(view: View){
    val intent=Intent()
    intent.action = "com.test.kotlin_test.MY_RECEIVER"
    intent.setPackage("com.test.kotlin_test")
    sendBroadcast(intent)
}

5.发送有序广播

fun sendOrBroadcast(view: View){
    val intent=Intent()
    intent.action = "com.test.kotlin_test.MY_RECEIVER"
    intent.setPackage("com.test.kotlin_test")
    sendOrderedBroadcast(intent,null)
}

发送有序广播只需要使用这个方法:sendOrderedBroadcast,第二个参数是跟权限相关的,可以暂时传null,接收者优先级高的先接收到有序广播,可以调用abortBroadcast()这个方法终止广播。其他的接收者就不能收到这个广播的。

<receiver
        android:name=".broadcast.OtherReceiver"
        android:enabled="true"
        android:exported="true">
    <intent-filter android:priority="100">
        <action android:name="com.test.kotlin_test.MY_RECEIVER"/>
    </intent-filter>
</receiver>

在intent-filter中可以配置priority 这个是广播接收者的优先级,默认是0,值越大优先级越高,取值范围是-1000,1000之间的数值。

6.广播的应用,强制退出登录功能

强制退出功能,可以使用广播的功能来实现,思路就是当需要强制退出的时候发送广播,接收到广播之后弹出弹窗,用户确认退出之后杀死所有的Activity并启动登录页面。

object ActivityController {
    private val activitys=ArrayList<Activity>()

    fun addActivity(activity: Activity){
        activitys.add(activity)
    }

    fun removeActivity(activity: Activity){
        activitys.remove(activity)
    }

    fun finishAll(){
        for (activity in activitys){
            if (!activity.isFinishing){
                activity.finish()
            }
        }
        activitys.clear()
    }
}

需要关闭所有的Activity,所以需要建立一个Activity的管理者,这个来管理Activity

open class BaseActivity :AppCompatActivity() {

    private lateinit var receiver:ForceOfflineReceiver

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        ActivityController.addActivity(this)
    }

    override fun onDestroy() {
        super.onDestroy()
        ActivityController.removeActivity(this)
    }

    override fun onResume() {
        super.onResume()
        val intentFilter=IntentFilter()
        intentFilter.addAction("com.test.kotlin_test.FORCE_OFFLINE")
        receiver=ForceOfflineReceiver()
        registerReceiver(receiver,intentFilter)
    }


    override fun onPause() {
        super.onPause()
        unregisterReceiver(receiver)
    }

    inner class ForceOfflineReceiver :BroadcastReceiver(){
        override fun onReceive(context: Context, intent: Intent?) {
            AlertDialog.Builder(context).apply {
                setTitle("注意")
                setMessage("你正在操作强制退出登录")
                setCancelable(false)
                setPositiveButton("ok"){_,_->
                    ActivityController.finishAll() //销毁所有的Activity
                    val intent=Intent(context,LoginActivity::class.java)
                    context.startActivity(intent)
                }
                show()
            }
        }

    }

定义一个baseActivity,让其他所有的Activity都继承这个,方便管理其他的页面,同时注意广播接收者的注册和反注册分别是在onResume和onPause声明周期里边,这样可以及时的注册和反注册广播,接收到广播之后弹出弹窗,用户点击确定后首先关闭所有Activity然后启动登录页面。

这样强制退出的功能就实现了。

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

推荐阅读更多精彩内容