Android广播机制
为了便于进行系统级别的消息通知,Android引入了一套广播消息机制。
Android中的广播机制十分灵活,每个程序都可以对自己感兴趣的广播进行注册。程序只会接收自己所关心的广播内容,这些广播内容可能来自系统,也可能来自其他应用程序。
发送广播:Intent;接收广播:广播接收器(Broadcast Receiver)。
广播类型分为两种:标准广播和有序广播
- 标准广播Normal broadcasts:一种完全异步执行的广播。广播发出后,所有广播接收器同一时刻接收到广播消息。
特点:效率高,无法被截断。 - 有序广播Ordered broadcasts:一种同步执行的广播。广播发出后,同一时刻只有一个广播接收器会受到这条广播消息。当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。
特点:广播接收器有优先级,优先级高的可以截断正在传递的广播。
接收系统广播
包括开机、电池电量变化、时间或时区发生改变等事件都会发出一条广播。
注册广播的方式一般有两种——在代码中注册(动态注册)和在AndroidManifest.xml中注册(静态注册)。
动态注册
创建一个广播接收器:创建一个继承自BroadcastReceiver
类的类,并重写父类的onReceive()
方法,写入处理逻辑。
使用动态注册监听网络变化,例子:
public class MainActivity extends AppCompatActivity {
private IntentFilter intentFilter;
private NetworkChangeReceiver networkChangeReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intentFilter = new IntentFilter();
intentFilter.addAction("android.net.comm.CONNECTIVITY_CHANGE");
networkChangeReceiver = new NetworkChangeReceiver();
registerReceiver(networkChangeReceiver, intentFilter);
}
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(networkChangeReceiver);
}
class NetworkChangeReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
ConnectivityManager connectionManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectionManager.getActiveNetworkInfo();
if(networkInfo != null && networkInfo.isAvailable()){
Toast.makeText(context, "network is available", Toast.LENGTH_SHORT).show();
}else{
Toast.makeText(context, "network is unavailable", Toast.LENGTH_SHORT).show();
}
}
}
}
- 创建一个IntentFilter实例,添加了一个值为
android.net.comm.CONNECTIVITY_CHANGE
的广播。需要监听什么,就在这里传入相应的action - 创建NetworkChangeReceiver的实例,调用
registerReceive()
方法进行注册,将两个实例都粗含进去 - 动态注册的广播接收器一定都要取消注册。一般可以在
onDestroy()
注销活动时调用unregisterReceiver()
方法来实现。 - 需要网络权限(写代码时就会提醒添加权限申请。)
《《《《《大无语,不知道为什么还是没过》》》》》
动态注册的特点:
- 优:自由控制注册与注销,灵活性强。
- 缺:必须在程序启动之后才能接收到广播。
静态注册
针对动态注册的缺点,可以使用静态注册方法。
以使用静态注册实现开机启动为例:
1.创建
右键com.example.broadcasttest包->新建->Other->Broadcast Receiver,弹出注册窗口。
- 选项Exported:是否允许这个广播接收器接收本程序以外的广播。
- 选项Enabled:是否启用这个广播接收器。
2.编辑
修改BootCompleteReceiver中的代码:
public class BootCompleteReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show();
}
}
3.注册
其实已经自动完成,打开AndroidManifest.xml文件可见。
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<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/AppTheme">
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
监听系统开机广播需要权限,在权限申请中完成添加。
可以看到已经自动生成的<receiver>
标签。系统启动后会发出一条值为android.permission.RECEIVE_BOOT_COMPLETED
的广播,因此需要在<intent-filter>
中添加相应的action。
要注意的一点是,在实际编写项目时不要在onReceive()方法中添加过多的逻辑或者任何耗时操作。因为广播接收器中不允许开启线程,当OnReceive()方法运行较长时间而没有结束时,程序就会报错。
发送自定义广播
挖坑 待续