Android学习笔记(十三)
Android中的广播机制
Android提供了一套完整的API,允许应用程序自由地发送和接受广播。
发送广播的方法借助于Intent,接受广播的方法需要广播接收器(BroadcastsReceiver)。
Android中的广播主要分为两种类型,标准广播和有序广播。
标准广播(Normal broadcasts)是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在
同一时刻接受到这条广播消息。
有序广播(Ordered broadcasts)是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播的消息,
当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。(这样,前面的广播接收器就可以截断正在传递的广播)
广播接收器可以自由地对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,
并在内部处理相应的逻辑。
注册广播的方式一般有两种,在代码中注册(也称为动态注册)和在AndroidManifest.xml中注册(也称为静态注册)。
广播接收器的创建:新建一个类,让其继承自BroadcastsReceiver,并重写父类的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的实例
intentFilter = new IntentFilter();
//广播接收器想要监听什么广播,就在这里添加相应的action
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
//创建一个NetworkChangeReceiver的实例
networkChangeReceiver = new NetworkChangeReceiver();
//调用registerReceiver()方法进行注册,将NetworkChangeReceiver的实例和IntentFilter的实例都传递进去
registerReceiver(networkChangeReceiver,intentFilter);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
//noinspection SimplifiableIfStatement
if (id == R.id.action_settings) {
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
protected void onDestroy() {
super.onDestroy();
//动态注册的广播接收器一定要取消注册
unregisterReceiver(networkChangeReceiver);
}
//定义一个内部类
class NetworkChangeReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//通过getSystemService()方法得到ConnectivityManager的实例,这是一个系统服务类,
// 专门用于管理网络连接的
ConnectivityManager connectivityManager =
(ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
//调用ConnectivityManager的getActiveNetworkInfo()方法可以得到NetworkInfo的实例
NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
//调用NetworkInfo的isAvailable()方法,就可以判断当前是否有网络了
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,这个类是继承自BroadcastsReceiver的,
并重写了onReceive()方法。这样每当系统的网络状态发生变化时,onReceive()中的方法就会执行。
在onReceive()方法中,首先通过getSystemService()方法得到ConnectivityManager的实例,这时一个系统服务类,专门用于
管理网络的。
然后调用ConnectivityManager的getActiveNetworkInfo()方法得到NetworkInfo的实例,接着调用NetworkInfo的isAvailable()方法,
就可以判断是否有网络了。
onCreate()方法中,首先创建了一个IntentFilter的实例,给它添加一个值为android.net.conn.CONNECTIVITY_CHANGE的action。网络
发生变化时,系统发出的正是一条值为android.net.conn.CONNECTIVITY_CHANGE的广播。
接下来创建一个NetworkChangeReceiver的实例,然后调用registerReceiver()方法进行注册,将NetworkChangeReceiver的实例和
IntentFilter的实例都传递进去,这样NetworkChangeReceiver就会收到所有值为android.net.conn.CONNECTIVITY_CHANGE的广播。
注意:动态注册的广播接收器一定要取消注册才行,这里在onDestroy()方法中通过调用unregisterReceiver()方法来实现。
代码如下所示:
protected void onDestroy() {
super.onDestroy();
//动态注册的广播接收器一定要取消注册
unregisterReceiver(networkChangeReceiver);
}
Android系统为了保证应用程序的安全性做了规定,如果程序要访问一些系统的关键性信息,必须要在AndroidManifest.xml文件中声明权限才可以。
例如下面的代码中就为查询系统的网络状态声明的权限,代码如下所示:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.mfeng.glh.broadcasttest" >
<!-- 添加查询系统网络状态的权限声明 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
运行程序,在系统设置中切换网络就可以收到广播消息了。
动态注册的广播可以自由的控制注册和注销,但是必须要程序启动以后才能接收广播。
如果想让程序在未启动的情况下接收广播,就要用的静态注册。
下面就使用静态注册的方法,接收一条开机广播,从而实现开机启动的功能。
新建一个BootCompleteReceiver类继承自BroadcastsReceiver,代码如下所示:
public class BootCompleteReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"Boot Complete",Toast.LENGTH_LONG).show();
}
}
这里就不在使用内部类的方式来定义广播接收器了。在onReceive()方法中,简单的弹出一段提示信息。
然后修改AndroidManifest.xml文件,代码如下:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.mfeng.glh.broadcasttest" >
<!-- 添加查询系统网络状态的权限声明 -->
<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:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<receiver android:name=".BootCompleteReceiver" >
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
</receiver>
</application>
</manifest>
上面的代码中,在<application>标签内出现了一个新的标签<receiver>,所有静态注册的广播接收器都是在这里进行注册的。
其中android:name指定具体注册哪一个广播接收器,然后在<intent-filter>标签中加入想要接收的广播就行了。
注意:监听系统开机广播也是需要声明权限的。
重新运行程序,就可以接收开机广播了。打开系统的应用程序管理界面,查看一下当前程序所拥有的权限,
如下图所示:

从图中可以看到,程序查看网络连接状态和开机启动的权限。
Android学习笔记(十三)的更多相关文章
- 【转】Pro Android学习笔记(十三):用户界面和控制(1):UI开发
目录(?)[-] UI开发 方式一通过XML文件 方式二通过代码 方式三XML代码 UI开发 先理清一些UI概念: view.widget.control:这三个名词其实没有什么区别,都是一个UI元素 ...
- 【转】 Pro Android学习笔记(二十):用户界面和控制(8):GridView和Spinner
目录(?)[-] GridView Spinner GridView GridView是网格状布局,如图所示.在了解ListView后,很容易了解GridView.下面是例子的XML文件. <? ...
- python3.4学习笔记(十三) 网络爬虫实例代码,使用pyspider抓取多牛投资吧里面的文章信息,抓取政府网新闻内容
python3.4学习笔记(十三) 网络爬虫实例代码,使用pyspider抓取多牛投资吧里面的文章信息PySpider:一个国人编写的强大的网络爬虫系统并带有强大的WebUI,采用Python语言编写 ...
- Android 学习笔记之Volley(七)实现Json数据加载和解析...
学习内容: 1.使用Volley实现异步加载Json数据... Volley的第二大请求就是通过发送请求异步实现Json数据信息的加载,加载Json数据有两种方式,一种是通过获取Json对象,然后 ...
- Android学习笔记进阶之在图片上涂鸦(能清屏)
Android学习笔记进阶之在图片上涂鸦(能清屏) 2013-11-19 10:52 117人阅读 评论(0) 收藏 举报 HandWritingActivity.java package xiaos ...
- android学习笔记36——使用原始XML文件
XML文件 android中使用XML文件,需要开发者手动创建res/xml文件夹. 实例如下: book.xml==> <?xml version="1.0" enc ...
- Android学习笔记之JSON数据解析
转载:Android学习笔记44:JSON数据解析 JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,采用完全独立于语言的文本格式,为Web应用开发提供了一种 ...
- java之jvm学习笔记十三(jvm基本结构)
java之jvm学习笔记十三(jvm基本结构) 这一节,主要来学习jvm的基本结构,也就是概述.说是概述,内容很多,而且概念量也很大,不过关于概念方面,你不用担心,我完全有信心,让概念在你的脑子里变成 ...
- udacity android 学习笔记: lesson 4 part b
udacity android 学习笔记: lesson 4 part b 作者:干货店打杂的 /titer1 /Archimedes 出处:https://code.csdn.net/titer1 ...
- Go语言学习笔记十三: Map集合
Go语言学习笔记十三: Map集合 Map在每种语言中基本都有,Java中是属于集合类Map,其包括HashMap, TreeMap等.而Python语言直接就属于一种类型,写法上比Java还简单. ...
随机推荐
- 《制造杀人犯/Making A Murder》中的疑点和感想
快马加鞭,一天之内看完了完整的10集,很震撼,在案件还存在诸多疑点的情况下.在还有如此多的细节无法确定的情况下,最后由12人组成的陪审团一致通过S.A的谋杀罪成立,尽管初次投票时有7票反对. ...
- Spring MVC之视图解析器和URL-Pattern的配置方案
上期讲解了第一入门案例之后接下来了解一下视图解析器与URL-Pattern的配置方案 先来说视图解析器,在上次博客文章中我们完成了入门案例,接下来我们就在上一个例子中完善一下体出视图解析器 <? ...
- powerdesigner 生成mysql PDM 的COMMENT注释
1powerdesigner 生成mysql PDM 的COMMENT注释 默认的pd没有生成注释,针对mysql5.0可以如下修改.在Database-->edit Current DBMS. ...
- ionic 跨页面传值的几种方法
1.使用AngularJS自带的$cacheFactory服务 $cacheFactory 从字面直译即为缓存工厂,可以用它来生成缓存对象,缓存对象以key-value的方式进行数据的存储,在整个应用 ...
- 使用Burpsuite抓取手机APP的HTTPS数据
1.所需条件 · 手机已经获取root权限 · 手机已经成功安装xposed框架 · 电脑一台 2.详细步骤 2.1 在手机上面安装xposed JustTrustMe JustTrustMe是一个去 ...
- js跳转传递参数
额,利用j获取了GridView中选中行数据后,通过JavaScript做跳转,传递参数的时候发现,当参数有中文的时候就会乱码, 当然出现这种情况的时候就需要对跳转的url进行编码 var urlX ...
- Mongodb和Hive详细对比
本文主要用于分析在大数据场景下Mongodb和Hive的优缺点: 支持的数据类型 支持的查询 支持的数据量 性能优化手段
- 运动规划 (Motion Planning): MoveIt! 与 OMPL
原创博文:转载请标明出处:http://www.cnblogs.com/zxouxuewei 最近有不少人询问有关MoveIt!与OMPL相关的话题,但是大部分问题都集中于XXX功能怎么实现,XXX错 ...
- 数据获取以及处理Beta版本展示
产品描述 这个产品的目的是为了学霸网站提供后台数据获取以及处理操作.在alpha阶段基本调通的基础至上,我们希望在bate版本中加入对于问答对的处理,图片的获取等功能. 预期目标 在alpha阶段,我 ...
- eclipse编辑器配置
1.添加行号 在侧边空白处右键 勾选如图 2.改字体 window -> preferences 字体的常用配置 Consolas有一个问题是中文字体难以看清 解决方式有两种:一.把字体设置为C ...