Android中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自系统的,也可能是来自其他应用程序的.
Android提供了一套完整的API,允许应用程序自由的发送和接收广播,
1.标准广播(Normal broadcasts)是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播消息,因此他们之间没有任何先后顺序可言,这种广播的效率比较高,但同时意味着它是无法被截断的,
2.有序广播(Ordered broadcasts)是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能接收到这条广播,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递,所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播的消息了.
动态注册监听网络变化
广播接收器可以对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够接收到该广播,并在内部处理相应的逻辑,注册广播一般有两种方式,在代码中注册和在AndroidManifest.xml中注册,其中前者也被称之为动态注册,后者被称为静态注册.
那么应该如何创建一个广播接收器呢?其实只需要新建一个类,让它继承自BroadcastReceiver,并重写父类的onReceive()方法就行了.这样当广播到来时,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.conn.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 connectivityManager = (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
if(networkInfo != null && networkInfo.isAvailable()){
Toast.makeText(context,"network is acailable",Toast.LENGTH_SHORT).show();
}else{
Toast.makeText(context,"network is unacailable",Toast.LENGTH_SHORT).show();
}
}
}
}
可以看到在MainActivity中定义了一个内部类NetworkChangeReceiver,这个类是继承自BroadcastReceiver的,并重写了onReceive()方法.这样每当网络状态发生变化的时候,onReceive()方法就会得到执行,这里只是简单的使用Toast提示了一段文本信息.
然后观察onCreate()方法,首先我们创建了一个IntentFilter的实例,并添加了一个值为android.net.conn.CONNECTIVITY_CHANGE的action,因为当网络发生变化的时候,系统发出的正是一条值为android.net.conn.CONNECTIVITY_CHANGE的广播,也就是说我们的广播接收器想要监听什么广播,就在这里添加相应的action,接下来创建一个NetworkChangeReceiver的实例,然后调用registerReceiver()方法进行注册,将NetworkChangeReceiver的实例和IntentFilter的实例都传进去,这样NetworkChangeReceiver就可以收到所有值为android.net.conn.CONNECTIVITY_CHANGE的广播了,也就实现了监听网络变化的功能,
最后要记得,动态注册的广播接收器一定都要取消注册才行,所以在onDestroy()方法中通过调用unregisterReceiver()方法来实现.
在onReceive()方法中,首先通过getSystemService()方法得到了ConnectivityManager的实例,这是一个系统服务类,专门用于管理网络连接的,然后调用它的getActiveNetworkInfo()方法可以得到NetworkInfo的实例,接着调用NetworkInfo的isAvailable()方法,就可以判断是否有网络,
Android系统为了保护用户的设备安全和隐私,做了严格的规定:如果程序需要进行一些对用户来说比较敏感的操作,就必须在配置文件中声明权限才可以,否则程序将直接崩溃,
这里应该加入访问网络的权限
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.broadcasttest">
<user-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
</manifest>
静态注册实现开机启动
动态注册的广播接收器可以自由的控制注册与注销,在灵活性方面有很大的优势,但是它也存在着一个缺点,即必须要在程序启动之后才能接收到广播,因为注册的逻辑是写在onCreate()方法中的,
静态注册可以让程序在未启动的情况下就能够接收到广播.
public class BootCompleteReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent){
Toast.makeText(context,"Boot Complete",Toast.LENGTH_SHORT).show();
}
}
静态的广播接收器一定要在AndroidManifest.xml文件中注册才能使用,
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.broadcasttest">
<user-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<user-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>
</application>
</manifest>
注意:
不要在onReceive()方法中添加过多的逻辑或者进行比较耗时的操作,因为在广播接收器中是不允许开启线程的,当onReceive()方法运行较长时间而没有结束时,程序就会报错.