一、广播简介
Android中的广播主要可以分为两种类型:标准广播和有序广播。
标准广播( Normal broadcasts)是一种完全异步执行的广播,在广播发出之后,所有的广 播接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何先后顺序可言。这种广播的效率会比较高,但同时也意味着它是无法被截断的。
标准广播的工作流程.png
有序广播( Ordered broadcasts)则是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播消息了。
有序广播的工作流程.png
二、广播的接收
1.动态注册监听网络变化
广播接收器可以自由地对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,并在内部处理相应的逻辑。注册广播的方式一般有两种,在代码中注册和在AndroidManifest.xml中注册,其中前者也被称为动态注册,后者也被称为静态注册。
那么该如何创建一个广播接收器呢?其实只需要新建一个类,让它继承自BroadcastReceiver,并重写父类的 onReceive()方法就行了。这样当有广播到来时, onReceive()方法就会得到执行,具体的逻辑就可以在这个方法中处理。
新建一个BroadcastTest项目,然后修改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) {
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();
}
}
}
}
可以看到,我们在 MainActivity中定义了一个内部类NetworkChangeReceiver,这个类是继承自 BroadcastReceiver的,并重写了父类的 onReceive()方法。这样每当网络状态发生变化时, onReceive()方法就会得到执行。
在 onReceive()方法中,首先通过 getSystemService()方法得到了 ConnectivityManager的实例,这是一个系统服务类,专门用于管理网络连接的。然后调用它的 getActiveNetworkInfo()方法可以得到 NetworkInfo的实例,接着调用 NetworkInfo的 isAvailable()方法,就可以判断出当前是否有网络了,最后我们通过 Toast的方式对用户进行提示。
然后观察 onCreate()方法,首先我们创建了一个 IntentFilter的实例,并给它添加了个值为 android.net.conn. CONNECTIVITY_CHANGE的 action,为什么要添加这个值呢?因为当网络状态发生变化时,系统发出的正是一条值为android.net.conn.CONNECTIVITY_CHANGE的广播,也就是说我们的广播接收器想要监听什么广播,就在这里添加相应的action。
接下来创建了ー个 NetworkChangeReceiver的实例,然后调用 registerReceiver()方法进行注册,将Network ChangeReceiver的实例和 IntentFilter的实例都传了进去,这样 NetworkChangeReceiver就会收到所有值为 android.net.conn.CONNECTIVITY_CHANGE的广播,也就实现了监听网络变化的功能。
最后要记得,动态注册的广播接收器一定都要取消注册才行,这里我们是在onDestroy()方法中通过调用 unregisterReceiver()方法来实现的。
另外,这里有非常重要的一点需要说明, Android系统为了保护用户设备的安全和隐私,做了严格的规定:如果程序需要进行一些对用户来说比较敏感的操作,就必须在配置文件中声明权限才可以,否则程序将会直接崩溃。比如这里访问系统的网络状态就是需要声明权限的。
打开AndroidManifest xml文件,在里面加入如下权限就可以访问系统网络状态了:
…
整体来说,代码还是非常简单的,现在运行一下程序。首先你会在注册完成的时候收到一条广播,然后按下Home键回到主界面(注意不能按Back建,否则 onDestroy()方法会执行),接着打开WLAN启动和禁用网络,你就会看到有 Toast提醒你网络发生了变化。
20190504_220455.gif
2.静态实现开机启动注册
动态注册的广播接收器可以自由地控制注册与注销,在灵活性方面有很大的优勢,但是它也存在着一个缺点,即必须要在程序启动之后才能接收到广播,因为注册的逻辑是写在onCreate()方法中的。那么有没有什么办法可以让程序在未启动的情况下就能接收到广播呢?这就需要使用静态注册的方式了。
这里我们准备让程序接收一条开机广播,当收到这条广播时就可以在onReceive()方法里执行相应的逻辑,从而实现开机启动的功能。可以使用 AndroidStudio提供的快捷方式来创建一个广播接收器
图片.png
右击 com. example. broadcasttest包→ New→ Other→Broadcast Receiver
图片.png
这里我们将广播接收器命名为 BootCompleteReceiver, Exported属性表示是否允许这个广播接收器接收本程序以外的广播, Enabled属性表示是否启用这个广播接收器。勾选这两个属性,点击 Finish完成创建。
然后修改 BootCompleteReceiver中的代码,如下所示
public class BootCompleteReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, “Boot Complete”, Toast.LENGTH_LONG).show();
}
}
代码非常简单,我们只是在onReceive()方法中使用 Toast弹出一段提示信息。
另外,静态的广播接收器一定要在AndroidManifest. xml文件中注册才可以使用,不过由于我们是使用 AndroidStudio的快捷方式创建的广播接收器,因此注册这一步已经被自动完成了。打开AndroidManifest.xml文件瞧一瞧,代码如下所示:
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
...
<receiver
android:name=".BootCompleteReceiver"
android:enabled="true"
android:exported="true">
</receiver>
</application>
可以看到,标签内出现了一个新的标签< receiver>,所有静态的广播接收器都是在这里进行注册的。它的用法其实和< activity>标签非常相似,也是通过 android: name来指定具体注册哪一个广播接收器,而 enabled和 exported属性则是根据我们刚才勾选的状态自动生成的。
不过目前 BootCompleteReceiver还是不能接收到开机广播的,我们还需要对 AndroidMainfest.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: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>
由于 Android系统启动完成后会发出一条值为android.intent. action.BOOT_COMPLETED的广播,因此我们在< intent- filter>标签里添加了相应的 action。另外,监听系统开机广播也是需要声明权限的,可以看到,我们使用 标签又加入了一条android.permission. RECEIVE_BOOT_COMPLETED权限。
现在重新运行程序后,我们的程序就已经可以接收开机广播了。将手机关闭并重新启动,在启动完成之后就会收到开机广播。
需要注意的是不要在onReceive()方法中添加过多的逻辑或者进行任何的耗时操作,因为在广播接收器中是不允许开启线程的,当onReceive()方法运行了较长时间而没有结束时,程序就会报错。因此广播接收器更多的是扮演一种打开程序其他组件的角色,比如创建一条状态兰通知,或者启动一个服务等。
三、发送自定义广播
1.发送标准广播
2.发送有序广播
四、使用本地广播
作者:zccon
链接:https://www.jianshu.com/p/a7466da45c6a
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。