在一个IP网络范围中,最大的ip地址是被保留用作广播地址来使用的,比如某个IP范围是192.168.0.xxx,子网掩码是255.255.255.0,那么这个网络的广播地址就是192.168.0.255.广播数据包会被发送到同一个网络上的所有端口,这样在该网络中的每台主机都会接收到这条广播.
1.广播机制简介:
发送广播方法是之前学的intent 接收广播的方法需要一个广播接收器(BroadcastReceiver)
什么是标准广播和有序广播?
标准广播是一种异步广播,即在广播发出后,几乎所有的广播接收器都会在同一时刻接收到这条广播信息,他们之间没有顺序可言,广播效率比较高,但是不能截断.
有序广播是一种同步执行的广播,在广播发出后,同一时刻只会有一个广播接收器接收到这条广播的消息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递.
2.接收系统广播
什么是系统广播?
系统广播比如手机开机之后会发出一条广播,电池电量发生状态变化会发出一条广播,时间或时区发生改变也会发出一条广播.
动态监听网络变化
注册广播的方式分为动态注册和静态注册
静态注册 在AndroidManifest.xml中注册;动态注册 在代码中注册.
前面我们说了应用程序发送广播和接收广播分别通过intent 和广播接收器(BroadcastReceiver),那么如何新建一个广播接收器呢??
只需要创建一个类,让他继承BoradcastReceiver,并重写BoradcastReceiver的子方法onReceive()即可.当有广播来时,onReceive()方法变回执行,具体的逻辑可以在这个方法中处理.
public class MainActivity extends AppCompatActivity {
private NetworkChangeReceiver networkChangeReceiver;
private IntentFilter intentFilter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
IntentFilter intentFilter=new IntentFilter();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
NetworkChangeReceiver 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
Toast.makeText(context,"network is changed",Toast.LENGTH_SHORT).show();}}
}
首先看我们创建的NetworkChangeReceiver这个广播接收器类,它继承了BroadcastReceiver这个类,并重写了这个类的onReceive方法,并在这个方法中执行Toast()操作.
再看onCreat()方法,在这个方法中我们创建了一个IntentFilter实例,并给它添加了一个值为android.net.conn.CONNECTIVITY_CHANGE的action,因为网络状态发生变化时,系统发出的正是一条值为android.net.conn.CONNECTIVITY_CHANGE的广播,也就是说我们的广播接收器想要监听什么广播,就在这里添加相应的action。(关于IntentFilter的用法详情请见http://blog.csdn.net/today520/article/details/7000048 ,总的来说就是一个过滤器可以认为是intent过滤器)然后再onCreat()方法中创建NetworkChangeReceiver的实例,最后调用registerReceiver方法进行注册,将NetworkChangeReceiver的实例和IntentFilter实例都传进去,这样NetworkChangeReceiver就会接收所有值为android.net.conn.CONNECTIVITY_CHANGE的广播,也就实现了监听网络变化的功能.
最后要记得,动态注册的广播接收器一定都要取消注册才行,这里我们是在。onDestroy()方法中通过调用unregisterReceiver()方法来实现的。(其中unregisterReceiver()方法中传入的是我们创建的这个NetworkChangeReceiver这个广播接收器)
如果我们觉得提醒网络变化不够人性化,那么我们可以对代码进行如下调整.
public class MainActivity extends AppCompatActivity{
........
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();}
}
}}
需要注意的是:首先通过getSystemService(Context.CONNECTIVITY SERVICE);方法获得 ConnectivityManager的实例,这个一个系统服务类(专门用于管理网络连接的,可以通过查其源代码发现返回的是Object对象,所以需要对其进行强制转型.),然后通过调用它的getActiveNetworkInfo();方法可以判断当前是否有网络,然后加一个if语句来判断,判断条件是其若不为空,且存在则....否则 ....
Android为了保护用户的隐私和设备安全,严格规定:如果程序需要进行一些对用户来说比较敏感的操作,就必须要在配置文件中声明权限才可以,否则程序会很容易崩溃.比如这里的访问系统的网络状态就是需要声明权限的.声明权限是在AndroidManifest.xml文件.
静态注册实现开机启动
动态注册广播接收器具有很大的灵活性,可是必须要在程序启动后才能接收的广播,因为注册逻辑是写在onCreat()方法中的,静态注册可以实现让程序在未启动的情况下就能接收广播.步骤如下右击。包--New---Other-Broadcast Receiver,会弹出一个窗口
可以看到,这里我们将广播接收器命名为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();
}}
另外,静态的广播接收器一定要在AndroidManifest.xml文件中注册才可以使用,不过由于我们是使用Android Studio的快捷方式创建的广播接收器,因此注册这一步已经被自动完成了。代开AndroidManifest.xml文件会看到一个<receiver>标签属性有name enabled expoorted等
它的用法其实和<activity>标签非常相似,也是通过android:name来指定具体注册哪一个广播接收器,而enabled和exported属性则是根据我们刚才勾选的状态自动生成的。还要再给AndroidManifest.xml文件添加
才能收到开机广播.
由于Android系统启动完成后会发出一条值为android.intent.action.B00T_COMPLETED的广播,因此我们在<intent-filter>标签里添加了相应的action。另外,监听系统开机广播也是需要声明权限的,可以看到,我们使用<user-permission>标签又加人了一条android.permission.RECEIVE_ BOOT COMPLETED权限.
发送自定义广播
上面的内容主要是通过广播接收器来接收系统广播,接下来我们学习一下如何在应用程序中发送自定义广播.
发送标准广播
发送广播之前首先定义一个广播接收器来接收次广播才行,不然发出去也是白发.新建一个MyBroadcastReceiver.
public class MyBroadcastReceiver extends BroadcastReceiver {
//------------发送自定义广播------------发送标准广播--------发送有序广播----
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).show();
//-------------发送有序广播中的广播截断-----------
//在设置了接收广播的优先权之后,那么MyBroadcastReceiver便可以自己选择是否允许广播继续传递了 //
通过abortBroadcast()方法 这个方法的意思是终止发送的意思
abortBroadcast();
} }
这里当MyBroadcastReceiver收到自定义广播时,就会弹出received in MyBroadcastReceiver的提示
在AndroidMinifest.xml文件中修改
让MyBroadcastReceiver接收一条值为com.exampte.broadcasttest.MY_BROADCAST的广播,因此待会儿在发送广播的时候,我们就需要发出这样的一条广播。
首先构造了一个Intent对象,并把要发送的广播的值传入,然后调用了Context的sendBroadcast()方法将广播发送出去,这样所有监听com.example.broadcasttest.MY_BROADCAST这条广播接收器都会收到消息.
在这里要和IntentFilter对比一下,在之前我们说了用intent发送消息,用广播接收器接收接收广播,IntentFilter是采用addAction()的方法将网络状态变化添加进去,Intent直接将要发送的广播的值传入其构造方法中,然后调用context的sendBroadcast()方法来发送广播,这样所有监听com.example.broadcasttest.MY_BROADCAST这条广播接收器(例如MyBroadcastReceiver)都会收到消息.
提一个问题:IntentFilter和AndroidMinifest.xml文件中intent-filter标签有什么关系???
发送有序广播和标准广播基本上一个套路,只不过在MainActivity中将sendBroadcast(intent)改成了sendOrderBroadcast(intent,null)
想要发送有序广播最好还是多建几个广播接收器,这样效果比较好.比如再建一个AnotherBroadcastReceiver,同样也是重写onReceive()方法,然后在AndroidMinifest.xml文件中修改<intent-filter>标签,将标签中的action的名字指定为com.example.broadcasttest.MY_BROADCAST因为只有将AnotherBroadcastReceiver也接收这条广播,现在运行BroadcastTest2项目将这个程序安装到模拟器上,然后重新回到BroadcastTest项目的主界面,并点击一下Send Broadcast按钮,就会分别弹出两次提示信息.
给广播接收器设定先后顺序在AndroidMinifest.xml中的intent-filter标签中加上 android:priorty="100"即可!(android:priorty属性给广播接收器设置了优先级,优先级比较高的广播接收器可以先收到广播)-----获得了接收广播的优先权之后 因为我们这次写的代码中MyBroadcastReceiver的优先级要高于AnotherBroadcastReceiver的优先级,所以在MyBroadCastReceiver.java中加上aborBroadcasr()方法(该方法的意思是终止发送广播即后面一个广播接收器AnotherBroadcastReceiver将收不到广播消息即我们所说的广播截断)
使用本地广播
系统全局广播容易引起安全性问题,因为发出的广播可以被任何应用程序接收到,并且我们也可以接受来自于其他任何应用程序的广播.
本地广播机制:使用这个广播机制发出去的广播只能够在应用程序的内部进行传递,并且广播接收器也只能接受来自本应用程序发出的广播
本地广播的用法和我们之前说的动态注册监听网络变化差不多,只不过本地广播用到了一个LocalBroadcastManager来对广播进行管理,并提供了发送广播和接收广播的方法.
我认为共同的套路都是新建一个广播接收器(LocalBroadcastReceiver),然后重写BroadcastReceiver的onReceive方法
新建一个IntentFilter对象然后用IntentFilter.addaction(要发送的广播的值)
IntentFilter和intent LocalBroadcastReceiver都要新建实例
无非就是多了:
1.采用LocalBroadcastManager.getInstance()方法来获得LocalBroadcastManager的实例
2.在构建Intent对象,并把要发送的广播的值植入inent对象之后,采用LocalBroadcastManager.sendBroadcast(intent)方法来发送广播
3.注册广播接收器的时候采用的是LocalBroadcastManager的registerReceiver()方法(方法中的参数是IntentFilter和localBroadcastReceiver)
4.取消注册广播接收器时是LocalBroadcastManager的unregisterReceiver()方法
本地广播无法通过静态注册的方式来接收!!!要注意!!