Android提供了一套完整的API,允许应用程序自由的发送和接受广播。
我们先大致的阐述下广播发送和接收的流程:
首先在需要发送信息的地方,把发送的信息和用于过滤的信息()装入一个Intent对象中,然后通过sendOrderBroadcast()和sendBroadcast()方法,把Intent对象以广播的方法发出。
当Intent发出后,所有已经注册的广播都会检测自己的IntentFilter是否与发送的Intent匹配,匹配则调用BroadCastReceiver中的onReceive()方法。
所以我们定义BroadCastReceiver都需要重写onReceive()。
广播的类型:
- 标准广播
这是一种完全异步执行的广播,在广播发出后,所有的广播接收器几乎都会在同一时刻接收到这条广播消息。 - 有序广播
这是一种同步执行的广播,在广播发出后,同一时刻只会有一个广播可接收到。所以这个广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播。
接收系统广播:
注册广播的方式有两种,在代码中注册称为动态注册,在AndroidManifest.xml称为静态注册。
动态注册
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);
initWindow();
}
public void initWindow(){
// 定义一个IntentFilter的实例
intentFilter = new IntentFilter();
// 为intentFilter添加action。广播接收器想要接受什么广播就在这添加相应的action
// 这里的action是网络状态发生改变时系统发出的。
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
networkChangeReceiver = new NetworkChangeReceiver();
// 调用registerReceiver()方法进行注册
registerReceiver(networkChangeReceiver,intentFilter);
}
@Override
protected void onDestroy() {
super.onDestroy();
// 在活动销毁时调用unregisterReceiver()方法注销广播接收器
unregisterReceiver(networkChangeReceiver);
}
//定义一个类继承BroadcastReceiver
class NetworkChangeReceiver extends BroadcastReceiver{
//重写onReceive()方法,当网路状态发生变化时调用
@Override
public void onReceive(Context context, Intent intent) {
// 通过getSystemService()方法得到ConnectivityManaget实例
// 这是一个系统服务类实例,专门用于管理网络连接
// 调用它的getActiveNetworkInfo()方法得到NetworkInfo实例
// 通过networkInfo.isAvailable()方法判断当前是否有网络
ConnectivityManager connectivityManager =
(ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectivityManager.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();
}
}
}
}
当然还不能忘记要在AndroidMainfest.xml中赋予权限
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
静态注册
动态注册必须要在程序启动之后才能接收到广播,因为注册是写在onCreate()方法中。
//BootCompleteReceiver.java
public class BootCompleteReceiver extends BroadcastReceiver {
public BootCompleteReceiver() {
}
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"Boot Complete",Toast.LENGTH_SHORT).show();
}
}
//AndroidMainfest.xml
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
.......
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
//监听系统开机广播
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
......
**需要注意的是,我们不要在onReceive()
方法中添加过多的逻辑或者进行任何的耗时操作,因为在广播中是不允许开启线程的,当onReceive()
运行时间过长程序会报错。
本地广播
因为上述的广播都是系统范围的广播,其可以接受系统内的所有广播,也会被系统中的所有应用接收其自身的广播,安全性无法保证。自此可以使用本地广播。本地广播的应用范围仅在应用内部,安全性高。
本地广播的应用与上述大致无差。只是使用LocalBroadCastManager本地广播管理器。
loaclBroadCastManager.register(broadCastReceiver);
loaclBroadCastManager.sendBroadCast(intent);
loaclBroadCastManager.unregister(broadCastReceiver);