广播机制介绍
- 广播类型
- 标准广播:是一种完全异步执行的广播,在广播发出以后,所有的广播接收器都会在同一个时刻接收这条广播,因此他们之间没有任何先后顺序可言,这种广播的效率比较高,同时意味这它是无法被拦截的
- 有序广播:则是一种同步执行的广播,在广播发出以后,同一时刻只有一个广播接收器能够接收到这条广播,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递,所以此时的广播是有先后顺序的,优先级高的广播接收器就可以先接收到广播,并且前面的广播接收器还可以阶段正在传递的广播,这样后买你的广播就没有办法接收到广播了
动态注册监听网络变化
广播接收器可以自由的对自己感兴趣的广播进行注册,这样当有相应的广播发出的时候,广播接收器就可以接收到该广播,并在内部处理相应的逻辑,注册广播的方式一般有两种,在代码中注册(动态注册)
和在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.main_activity);
// 创建一个实例
intentFilter = new IntentFilter();
// 添加一个action,这个是网络变化的时候的广播
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
networkChangeReceiver = new NetworkChangeReceiver();
// 进行注册,这样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()
方法,每当网络状态发生变化的时候,这个方法就会得到执行,用Toast提示一段文本信息 - 在onCreat()方法中,首先创造一个
IntentFiter
实例,并给他添加了action,添加的值正是当网络发生变化的时候系统发出的广播,我们的广播接收器想要监听什么样的广播,就在这里添加相应的action - 接下来创建NetworkChangeReceiver实例,然后调用
registerReceiver()
方法进行注册,把NetworkChangeReceiver的实例和intentFilter的实例都传进去,这样就实现了监听网络变化的功能 - 最后,动态注册广播的时候一定要取消注册,在onDestroy()中调用
unregisterReceiver()
方法来实现 -
运行程序,按下Home键回到主页面(此时不能按Back键,否则会执行onDestroy()中的方法),打开设置页面,点击数据使用详情的界面,按数据开关,这个是时候,就会看到Toast提醒的网络变化
- 此时的网络提醒还不够人性化,继续修改,其他的保持不变
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,"网络 开",Toast.LENGTH_LONG).show();
}else{
Toast.makeText(context,"网络 关",Toast.LENGTH_LONG).show();
}
}
}
- 首先通过
getSystemService(Context.CONNECTIVITY_SERVICE)
,得到ConnectivityManager的实例,这是一个系统服务类,专门用于管理网络连接,然后调用getActiveNetworkInfo()
;方法可以获取到NetworkInfo的实例,接着调用networkInfo.isAvailable()
方法就可以判断当前是否有网络 - 这个时候还必须的声明权限,否则程序就会直接崩溃,在AndroidManifest.xml中
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.md.bb">
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
...
</manifest>
-
重新运行程序,按Home,设置,移动网络,开关数据
静态注册实现开机启动
动态注册在广播接收器可以自由控制注册与注销,在灵活性方面有很大的优势。但是也有缺点,就是必须要在程序启动之后才能够接收到广播,因为注册是写在onCreate()中的,静态注册可以在程序没有启动的情况下就可以接收到广播
-
准备让程序接收一条开机的广播,当接收到这个广播的时候在onReceive()方法里执行响应的逻辑,右键 com.example.包,new Other Broadcast
可以看到广播接收器的名字,Exported属性表示是否允许这个广播接收本程序以外的广播,Enabled属性表示是否启动这个广播接收器,勾选上,Finish
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的快捷方式创建的广播接收器,所以这一步被系统完成了,打开这个文件
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.md.bb">
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<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">
</receiver>
</application>
</manifest>
- 可以看到在application中出现了一个新的标签receiver,所有静态的广播接收都是在这里注册的,目前还是不能接收到开机广播,还要对AndroidManifest.xml进行修改
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.md.bb">
<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>
</application>
</manifest>
- 由于Android系统在启动完成后会发出一个值为
android.intent.action.BOOT_COMPLETED
的广播因此我们在receiver/的intent-filter标签中添加了action,另外监听系统开机广播也需要声明权限,在uses-permission标签中添加了android.permission.RECEIVE_BOOT_COMPLETED
权限 - 此时运行程序就可以接收开机广播了,关闭模拟器并重新启动就可以接收了