《第一行代码》学习笔记:接受系统广播
Android中的广播主要可以分为两种类型:标准广播和有序广播。
- 标准广播 (Normal broadcasts)是一种完全异步执行的广播,在广播发出之后,所有的广播
接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何先后顺序可言。
这种广播的效率会比较高,但同时也意味着它是无法被截断的。 - 有序广播 (Ordered broadcasts)则是一种同步执行的广播,在广播发出之后,同一时刻只
会有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑执行完毕后,广
播才会继续传递。所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以
先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接
收器就无法收到广播消息了。
注册广播的方式一般有两种,在代码中注册和在AndroidManifest.xml中注册,其中前者也被称为动态注册,后者也被称为静态注册。
一、动态注册监听网络变化
要创建一个广播接收器:只需要新建一个类,让它继承自BroadcastReceiver ,并重写父类的onReceive() 方法就行了。这样当有广播到来时,onReceive() 方法就会得到执行,具体的逻辑就可以在这个方法中处理。
新建一个项目,然后修改MainActivity中的代码,如下所示:
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) {
Toast.makeText(context, "network changes", Toast.LENGTH_SHORT).show();
}
}
}
在MainActivity中定义了一个内部类NetworkChangeReceiver ,这个类是继承自BroadcastReceiver 的,并重写了父类的onReceive() 方法。这样每当网络状态发生变化时,onReceive() 方法就会得到执行。然后观察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() 方法来实现的。
为了准确地告诉用户当前是有网络还是没有网络,因此我们还需要对上面的代码进行进一步的优化。修改MainActivity中的内部类代码:
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();
}
}
}
在onReceive() 方法中,首先通过getSystemService() 方法得到了ConnectivityManager 的实例,这是一个系统服务类,专门用于管理网络连接的。然后调用它的getActiveNetworkInfo() 方法可以得到NetworkInfo 的实例,接着调用NetworkInfo 的isAvailable() 方法,就可以判断出当前是否有网络了,最后还是通过Toast的方式对用户进行提示。要在AndroidManifest.xml文件,在里面加入如下权限才可以访问系统网络状态:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.broadcasttest">
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
...
</manifest>
二、静态注册实现开机启动
静态注册可以让程序在未启动的情况下就能接收到广播。这里我们准备让程序接收一条开机广播,当收到这条广播时就可以在onReceive() 方法里执行相应的逻辑,从而实现开机启动的功能。可以使用Android Studio提供的快捷方式来创建一个广播接收器,右击com.example.broadcasttest包→New→Other→Broadcast Receiver,Exported 属性表示是否允许这个广播接收器接收本程序以外的广播,Enabled 属性表示是否启用这个广播接收器。然后修改广播接收器代码:
public class BootCompleteReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show();
}
}
另外,静态的广播接收器一定要在AndroidManifest.xml文件中注册才可以使用,不过由于我们是使用Android Studio的快捷方式创建的广播接收器,因此注册这一步已经被自动完成了。在<receiverAndroidManifest.xml文件中的<application>
内,有
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
</receiver>
由于Android系统启动完成后会发出一条值为android.intent.action.BOOT_COMPLETED
的广播,因此我们在<intent-filter>
标签里添加了相应的action。另外,监听系统开机广播也是需要声明权限的,需要在 标签又加入了一条android.permission.RECEIVE_BOOT_COMPLETED 权限。
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
需要注意的是,不要在onReceive() 方法中添加过多的逻辑或者进行任何的耗时操作,因为在广播接收器中是不允许开启线程的,当onReceive() 方法运行了较长时间而没有结束时,程序就会报错。因此广播接收器更多的是扮演一种打开程序其他组件的角色,比如创建一条状态栏通知,或者启动一个服务等。
本文地址:https://blog.csdn.net/weixin_44267444/article/details/107122349