一、广播机制

为了便于进行系统级别的消息通知,Android引入了一套广播消息机制。

每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会收到自己所关心的广播内容,这些广播可能是来自于系统的,也可能是来自于其他应用程序的。Android提供了一套完整的API,允许应用程序自由地发送和接收广播。

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


二、标准广播

标准广播(normal broadcasts)是一种完全异步执行的广播,在广播发出之后,所有的BroadcastReceiver几乎都会在同一时刻接收到收到这条广播消息,因此它们之间没有任何先后顺序可言。这种广播的效率会比较高,但同时也意味着它是无法被截断的。示意图如下:

Android第七课-----广播_监听器


三、有序广播

有序广播(ordered broadcasts)是一种同步执行的广播,在广播发出之后,同一时刻只会有一个BroadcastReceiver能够收到这条广播消息,当这个BroadcastReceiver中的逻辑执行完毕后,广播才会继续传递。所以此时的BroadcastReceiver是有先后顺序的,优先级高的BroadcastReceiver就可以先收到广播消息,并且前面的BroadcastReceiver还可以截断正在传递的广播,这样后面的BroadcastReceiver就无法收到广播消息了。示意图如下:

Android第七课-----广播_kotlin_02


四、接收系统广播

开发者可以根据自己感兴趣的广播,自由地注册BroadcastReceiver,这样当有相应的广播发出时,相应的BroadcastReceiver就能够收到该广播,并可以在内部进行逻辑处理。

注册BroadcastReceiver的方式一般有两种:在代码中注册和在AndroidManifest.xml中注册。其中前者也被称为动态注册,后者也被称为静态注册。


动态注册监听时间变化

Android第七课-----广播_监听器_03

package com.example.helloworld

import android.content.BroadcastReceiver
import android.content.Context
import android.support.v7.app.AppCompatActivity
import android.os.Bundle

import android.widget.* //引入widget组件包
import android.content.Intent
import android.content.IntentFilter
import android.net.Uri

class MainActivity : AppCompatActivity() {

lateinit var timeChangeReceiver: BroadcastReceiver

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

//匹配布局里的按钮
val button = findViewById<Button>(R.id.button)
//给按钮添加监听器
button.setOnClickListener{
val intent = Intent(MainActivity@this,MyActivity::class.java)
startActivity(intent)
}

//创建广播监听器
timeChangeReceiver = TimeChangeReceiver()
//创建内容过滤器
val intentFilter = IntentFilter()
intentFilter.addAction("android.intent.action.TIME_TICK")
//注册广播监听器
registerReceiver(timeChangeReceiver,intentFilter)
}

//程序销毁时,撤销注册了的广播监听器
override fun onDestroy() {
super.onDestroy()
unregisterReceiver(timeChangeReceiver)
}

//内部类
inner class TimeChangeReceiver : BroadcastReceiver() {
//重写onReceive,监听到广播的处理函数
override fun onReceive(context: Context?, intent: Intent?) {
Toast.makeText(context,"Time has changed",Toast.LENGTH_SHORT).show()
}
}
}

Android第七课-----广播_ide_04


静态注册实现开机启动

Android第七课-----广播_监听器_05

package com.example.helloworld

import android.content.BroadcastReceiver
import android.content.Context
import android.support.v7.app.AppCompatActivity
import android.os.Bundle

import android.widget.* //引入widget组件包
import android.content.Intent

class MainActivity : AppCompatActivity() {

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

//匹配布局里的按钮
val button = findViewById<Button>(R.id.button)
//给按钮添加监听器
button.setOnClickListener{
val intent = Intent(MainActivity@this,MyActivity::class.java)
startActivity(intent)
}

}
}

class BootCompleteReceiver : BroadcastReceiver(){
override fun onReceive(context: Context?, intent: Intent?) {
Toast.makeText(context,"Boot Complete",Toast.LENGTH_LONG).show()
}
}

Android第七课-----广播_ide_06

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.helloworld">

<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.HelloWorld">
<activity
android:name=".MyActivity"
android:exported="false" />
<activity
android:name=".MainActivity"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<receiver android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
</application>

</manifest>

五、发送自定义广播

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


发送标准广播

点击按钮发送广播,然后自己接收广播,使用Toast提示;

Android第七课-----广播_ide_07

package com.example.helloworld

import android.content.BroadcastReceiver
import android.content.Context
import android.support.v7.app.AppCompatActivity
import android.os.Bundle

import android.widget.* //引入widget组件包
import android.content.Intent
import android.content.IntentFilter
import android.net.Uri

class MainActivity : AppCompatActivity() {

//内部类
inner class SelfEventReceiver : BroadcastReceiver() {
//重写onReceive,监听到广播的处理函数
override fun onReceive(context: Context?, intent: Intent?) {
Toast.makeText(context,"收到自定义广播",Toast.LENGTH_SHORT).show()
}
}

lateinit var selfEventReceiver : BroadcastReceiver //lateinit表示待会初始化,类似于C的前置声明

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

//匹配布局里的按钮
val button = findViewById<Button>(R.id.button)
//给按钮添加监听器
button.setOnClickListener{
val intent = Intent("com.example.broadcasttest.MY_BROADCAST")
intent.setPackage(packageName)
sendBroadcast(intent)
}

//创建广播监听器
selfEventReceiver = SelfEventReceiver()
//创建内容过滤器
val intentFilter = IntentFilter()
intentFilter.addAction("com.example.broadcasttest.MY_BROADCAST")
//注册广播监听器
registerReceiver(selfEventReceiver,intentFilter)
}

//程序销毁时,撤销注册了的广播监听器
override fun onDestroy() {
super.onDestroy()
unregisterReceiver(selfEventReceiver)
}
}

Android第七课-----广播_android studio_08


发送有序广播
和标准广播不同,有序广播是一种同步执行的广播,并且是可以被截断的。

发送有序广播只需要改动一行代码,即将​​sendBroadcast()​​​方法改成​​sendOrderedBroadcast()​​方法。

Android第七课-----广播_kotlin_09