频道栏目
首页 > 程序开发 > 移动开发 > Android > 正文
Android Broadcast Receiver机制的实例详解
2018-06-12 12:01:34      个评论    来源:Hy的博客  
收藏   我要投稿

《第一行代码》

一、广播机制简介

为什么说 Android 中的广播机制更加灵活呢?这是因为 Android 中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自于系统的,也可能是来自于其他应用程序的。 Android 提供了一套完整的API,允许应用程序自由地发送和接收广播。发送广播的方法其实之前稍微有提到过一下,如果你记性好的话可能还会有印象,就是借助我们第 2 章学过的 Intent。而接收广播的方法则需要引入一个新的概念,广播接收器(Broadcast Receiver)。

广播接收器的具体用法将会在下一节中做介绍,这里我们先来了解一下广播的类型。Android 中的广播主要可以分为两种类型,标准广播和有序广播。

标准广播(Normal broadcasts) 是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何先后顺序可言。这种广播的效率会比较高,但同时也意味着它是无法被截断的。标准广播的工作流程如图 5.1 所示。

这里写图片描述

有序广播(Ordered broadcasts) 则是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播消息了。有序广播的工作流程如图 5.2 所示。<喎"/kf/ware/vc/" target="_blank" class="keylink">vcD4NCjxwPjxpbWcgYWx0PQ=="这里写图片描述" src="/uploadfile/Collfiles/20180612/201806120931592.png" title="\" />

二、接收系统广播

Android 内置了很多系统级别的广播,我们可以在应用程序中通过监听这些广播来得到各种系统的状态信息。比如手机开机完成后会发出一条广播,电池的电量发生变化会发出一条广播,时间或时区发生改变也会发出一条广播等等。如果想要接收到这些广播,就需要使用广播接收器,下面我们就来看一下它的具体用法。

1、动态注册监听网络变化

广播接收器可以自由地对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,并在内部处理相应的逻辑。注册广播的方式一般有两种,在代码中注册和在 AndroidManifest.xml 中注册,其中前者也被称为动态注册,后者也被称为静态注册。

那 么 该 如 何 创 建 一 个 广 播 接 收 器 呢 ? 其 实 只 需 要 新 建 一 个 类 , 让 它 继 承 自BroadcastReceiver, 并重写父类的 onReceive()方法就行了。这样当有广播到来时,onReceive()方法就会得到执行,具体的逻辑就可以在这个方法中处理。

那我们就先通过动态注册的方式编写一个能够监听网络变化的程序,借此学习一下广播接收器的基本用法吧。新建一个 BroadcastTest 项目,然后修改 MainActivity 中的代码,如下所示:

public class MainActivity extends Activity {
 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()方法就会得到执行,这里只是简单地使用 Toast 提示了一段文本信息。

然后观察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()方法来实现的。整体来说,代码还是非常简单的,现在运行一下程序。首先你会在注册完成的时候收到一条广播,然后按下 Home键回到主界面(注意不能按Back键,否则onDestroy()方法会执行),接着按下Menu键→System settings→Data usage 进入到数据使用详情界面,然后尝试着开关 Mobile Data 来启动和禁用网络,你就会看到有 Toast 提醒你网络发生了变化。

不过只是提醒网络发生了变化还不够人性化,最好是能准确地告诉用户当前是有网络还是没有网络,因此我们还需要对上面的代码进行进一步的优化。修改 MainActivity 中的代码,如下所示:

public class MainActivity extends Activity {
 ……
 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 的方式对用户进行提示。

另外,这里有非常重要的一点需要说明, Android 系统为了保证应用程序的安全性做了规定,如果程序需要访问一些系统的关键性信息,必须在配置文件中声明权限才可以,否则程序将会直接崩溃,比如这里查询系统的网络状态就是需要声明权限的。打开AndroidManifest.xml 文件,在里面加入如下权限就可以查询系统网络状态了:


 
 
 ……

2、静态注册实现开机启动

动态注册的广播接收器可以自由地控制注册与注销,在灵活性方面有很大的优势,但是它也存在着一个缺点,即必须要在程序启动之后才能接收到广播,因为注册的逻辑是写在onCreate()方法中的。那么有没有什么办法可以让程序在未启动的情况下就能接收到广播呢?这就需要使用静态注册的方式了。

这里我们准备让程序接收一条开机广播,当收到这条广播时就可以在onReceive()方法里执行相应的逻辑,从而实现开机启动的功能。新建一个 BootCompleteReceiver 继承自BroadcastReceiver,代码如下所示:

public class BootCompleteReceiver extends BroadcastReceiver {
 @Override
 public void onReceive(Context context, Intent intent) {
  Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show();
 }
}

可以看到,这里不再使用内部类的方式来定义广播接收器,因为稍后我们需要在AndroidManifest.xml 中将这个广播接收器的类名注册进去。在 onReceive()方法中,还是简单地使用 Toast 弹出一段提示信息。

然后修改 AndroidManifest.xml 文件,代码如下所示:


 ……
 
 
 
  ……
  

 
  
 
 

终于,< application>标签内出现了一个新的标签< receiver>,所有静态注册的广播接收器都是在这里进行注册的。它的用法其实和< activity>标签非常相似,首先通过android:name 来指定具体注册哪一个广播接收器,然后在< intent-filter>标签里加入想要接 收 的 广 播 就 行 了 , 由 于 Android 系 统 启 动 完 成 后 会 发 出 一 条 值 为android.intent.action.BOOT_COMPLETED 的广播,因此我们在这里添加了相应的action。

另外,监听系统开机广播也是需要声明权限的,可以看到,我们使用< uses-permission>标签又加入了一条 android.permission.RECEIVE_BOOT_COMPLETED 权限。现在重新运行程序后,我们的程序就已经可以接收开机广播了,首先打开到应用程序管理界面来查看一下当前程序所拥有的权限。在桌面按下Menu键→System settings→

Apps,然后点击 BroadcastTest,如图 5.5 所示。

这里写图片描述

可以看到,我们的程序目前拥有访问网络状态和开机自动启动的权限。然后将模拟器关闭并重新启动,在启动完成之后就会收到开机广播了,如图 5.6 所示。

这里写图片描述

到目前为止,我们在广播接收器的 onReceive()方法中都只是简单地使用 Toast 提示了一段文本信息,当你真正在项目中使用到它的时候,就可以在里面编写自己的逻辑。需要注意的是,不要在 onReceive()方法中添加过多的逻辑或者进行任何的耗时操作,因为在广播接收器中是不允许开启线程的,当 onReceive()方法运行了较长时间而没有结束时,程序就会报错。因此广播接收器更多的是扮演一种打开程序其他组件的角色,比如创建一条状态栏通知,或者启动一个服务等,这几个概念我们会在后面的章节中学到。

3、发送自定义广播

现在你已经学会了通过广播接收器来接收系统广播,接下来我们就要学习一下如何在应用程序中发送自定义的广播。前面已经介绍过了,广播主要分为两种类型,标准广播和有序广播,在本节中我们就将通过实践的方式来看下这两种广播具体的区别。

1)发送标准广播

在发送广播之前,我们还是需要先定义一个广播接收器来准备接收此广播才行, 不然发出去也是白发。因此新建一个 MyBroadcastReceiver 继承自 BroadcastReceiver,代码如下所示:

public class MyBroadcastReceiver extends BroadcastReceiver {
 @Override
 public void onReceive(Context context, Intent intent) {
  Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();
 }
}

这里 当 MyBroadcastReceiver 收 到 自 定 义 的 广 播 时 , 就 会 弹 出 received in MyBroadcastReceiver 的提示。然后在 AndroidManifest.xml 中对这个广播接收器进行注册:


 ……
 
  ……
  
  


  
 

可以看到,这里让MyBroadcastReceiver接收一条值为com.example.broadcasttest.MY_BROADCAST的广播,因此待会儿在发送广播的时候,我们就需要发出这样的一条广播。

public class MainActivity extends Activity {
 ……
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  Button button = (Button) findViewById(R.id.button);
  button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
 Intent intent = new Intent("com.example.broadcasttest.MY_BROADCAST");
 sendBroadcast(intent);
}
  });
  ……
 }
 ……
}

可以看到,我们在按钮的点击事件里面加入了发送自定义广播的逻辑。首先构建出了一个Intent对象,并把要发送的广播的值传入,然后调用了Context的sendBroadcast()方法将广播发送出去,这样所有监听com.example.broadcasttest.MY_BROADCAST这条广播的广播接收器就会收到消息。此时发出去的广播就是一条标准广播。

2)发送有序广播

广播是一种可以跨进程的通信方式,这一点从前面接收系统广播的时候就可以看出来了。因此在我们应用程序内发出的广播,其他的应用程序应该也是可以收到的。为了验证这一点,我们需要再新建一个 BroadcastTest2 项目。

将项目创建好之后,还需要在这个项目下定义一个广播接收器,用于接收上一小节中的自定义广播。新建 AnotherBroadcastReceiver 继承自 BroadcastReceiver,代码如下所示:

public class AnotherBroadcastReceiver extends BroadcastReceiver {
 @Override
 public void onReceive(Context context, Intent intent) {
  Toast.makeText(context, "received in AnotherBroadcastReceiver", Toast.LENGTH_SHORT).show();
 }
}
点击复制链接 与好友分享!回本站首页
上一篇:3D框架Sceneform关键概念及ARCore在Android Studio环境下的简单入门教程
下一篇:简单谈谈Android中的热修复原理
相关文章
图文推荐

关于我们 | 联系我们 | 广告服务 | 投资合作 | 版权申明 | 在线帮助 | 网站地图 | 作品发布 | Vip技术培训 | 举报中心

版权所有: 红黑联盟--致力于做实用的IT技术学习网站