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然后启动登录页面。

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

源码地址

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值