Android学习笔记(十三)——广播机制

   //此系列博文是《第一行Android代码》的学习笔记,如有错漏,欢迎指正!

  Android 中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自于系统的,也可能是来自于其他应用程序的。Android 提供了一套完整的 API,允许应用程序自由地发送和接收广播。

一、广播的类型:

  Android 中的广播主要可以分为两种类型,标准广播和有序广播。

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

  2)有序广播(Ordered broadcasts)则是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播消息了。

二、接收系统广播:

  Android 内置了很多系统级别的广播,我们可以在应用程序中通过监听这些广播来得到各种系统的状态信息。比如手机开机完成后会发出一条广播,电池的电量发生变化会发出一
条广播,时间或时区发生改变也会发出一条广播等等。如果想要接收到这些广播,就需要使用广播接收器。广播接收器可以自由地对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,并在内部处理相应的逻辑。注册广播的方式一般有两种,在代码中注册和在 AndroidManifest.xml 中注册, 其中前者也被称为动态注册, 后者也被称为静态注册。

  1)动态注册:
  我们使用动态注册来实现这样的一个功能:提醒用户网络是否处于开启状态,代码如下:

 1 public class MainActivity extends AppCompatActivity {
 2
 3     private IntentFilter intentFilter;
 4     private NetworkChangeReceiver networkChangeReceiver;
 5     @Override
 6     protected void onCreate(Bundle savedInstanceState) {
 7         super.onCreate(savedInstanceState);
 8         setContentView(R.layout.activity_main);
 9         intentFilter = new IntentFilter();
10         intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
11         networkChangeReceiver = new NetworkChangeReceiver();
12         registerReceiver(networkChangeReceiver, intentFilter);
13     }
14     @Override
15     protected void onDestroy() {
16         super.onDestroy();
17         unregisterReceiver(networkChangeReceiver);
18     }
19     class NetworkChangeReceiver extends BroadcastReceiver {
20         @Override
21         public void onReceive(Context context, Intent intent) {
22             ConnectivityManager connectionManager = (ConnectivityManager)
23                     getSystemService(Context.CONNECTIVITY_SERVICE);
24             NetworkInfo networkInfo = connectionManager.getActiveNetworkInfo();
25             if (networkInfo != null && networkInfo.isAvailable()) {
26                 Toast.makeText(context, "network is available",
27                         Toast.LENGTH_SHORT).show();
28             } else {
29                 Toast.makeText(context, "network is unavailable",
30                         Toast.LENGTH_SHORT).show();
31             }
32         }
33     }
34 }

  我们在 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()方法进行注册,将 NetworkChangeReceiver 的实例和 IntentFilter 的实例都传了进去,这样NetworkChangeReceiver就会收到所有值为android.net.conn.CONNECTIVITY_CHANGE的广播,也就实现了监听网络变化的功能。

  最后要记得, 动态注册的广播接收器一定都要取消注册才行,这里我们是在 onDestroy()方法中通过调用 unregisterReceiver()方法来实现的。

  另外,这里有非常重要的一点需要说明,这里查询系统的网络状态就是需要声明权限的,否则程序将会直接崩溃,声明的话只需在manifest文件假如下列语句即可: 

1 <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>

  效果如下:

  

  2)静态注册:

  动态注册的广播接收器可以自由地控制注册与注销,在灵活性方面有很大的优势,但是它必须要在程序启动之后才能接收到广播,假如我们想让程序开机时就相应呢?这时候我们就可以使用静态注册了:

  (1)首先新建一个 BootCompleteReceiver 继承自BroadcastReceiver,代码如下:

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

  在OnRecieve()方法中我们只是简单地弹出一条文本。

  (2)在manifest文件中取得权限和声明广播,代码如下:

1 <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

   监听系统开机广播也是需要声明权限的,我们使用<uses-permission>标签又加入了一条 android.permission.RECEIVE_BOOT_COMPLETED 权限。接着我们在application标签下插入下面代码:

1 <receiver android:name=".BootCompleteReceiver" >
2     <intent-filter>
3           <action android:name="android.intent.action.BOOT_COMPLETED" />
4     </intent-filter>
5 </receiver>

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

  重新启动手机后,程序就会在屏幕中弹出文本:Boot Complete。

  //End.

时间: 2024-09-30 21:29:47

Android学习笔记(十三)——广播机制的相关文章

九、Android学习第八天——广播机制与WIFI网络操作(转)

(转自:http://wenku.baidu.com/view/af39b3164431b90d6c85c72f.html) 九.Android学习第八天——广播机制与WIFI网络操作 今天熟悉了Android中的广播机制与WIFI网络的一些基本操作,总结如下: Android的广播机制 我们知道广播机制中,发送方不会关心接收方时候接收到数据或者如何去处理数据. 这里总结下Android中BroadcastReceiver的注册方法: (一)在应用程序中进行注册 (二)在Manifest.xml

Android 学习笔记 BroadcastReceiver广播...

PS:不断提升自己,是件好事... 学习内容: 1.BroadcastReceiver的使用.. 2.通过BroadcastReceiver去启动Service... 1.BroadcastReceiver...   广播接收者,用来接收系统和应用中的广播...Android广播机制用的地方还是很多的,比如说:当网络状态发生改变的时候,通过接收这条广播就能及时做出相应的响应,然后完成一些操作..这就是广播的机制...它自身并不实现图形用户界面...只是一个触发性的操作...   使用广播是需要几

Android学习总结——本地广播机制

为了简单解决广播的安全性问题,Android引入了一套本地广播机制,使用这个机制发出的广播只能在程序的内部进行传递,只能接受来自本应用程序发出的广播.否则当我们发送一些携带关键数据的广播可能被截获,一些恶意程序也可能向我们的广播接收器中发送各种垃圾广播. 用法如下:(主要就是使用LocalBroadcastManager来对广播进行管理,并提供注册广播和发送广播的方法). package com.example.xch.localbroadcast; import android.content

Android学习笔记之消息机制

Android的消息机制主要是指Handler的运行机制以及Handler所附带的MessageQueue和Looper的工作过程.   1.为什么要使用Handler? Android规定访问UI只能在主线程中进行,如果在子线程中访问UI,则会抛出异常.同时Android建议不能在主线程中进行耗时的操作,不然会导致程序无法响应即ANR.因此耗时的工作只能交给子线程去做,而子线程却不能直接访问UI,为了解决这个矛盾,Android提供了Handler.Handler的主要作用是将某个任务切换到某

Android学习笔记之广播意图及广播接收者MyBroadcastReceiver、Broadcast

(1)第一种使用xml文件进行注册 布局文件,添加一个button点击的时候进行广播 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height

Android学习笔记十三.深入理解fragment(一)

深入理解fragment(一) Fragment是Android3.0引入的新API,可以把Fragment想成Activity中的模块,这个模块有自己的布局,有自己的生命周期,单独处理自己的输入,在Activity运行的时候可以加载或者移除Fragment模块. 可以把Fragment设计成可以在多个Activity中复用的模块,当开发的应用程序同时适用于平板电脑和手机时,可以利用Fragment实现灵活的布局,改善用户体验. 一.Fragment的特征 1.Fragment总是作为Activ

十三、Android学习笔记_Andorid控件样式汇总

<!-- 设置activity为透明 --> <style name="translucent"> <item name="android:windowBackground">@color/translucent</item> <item name="android:windowIsTranslucent">true</item> <item name="an

十四、Android学习笔记_Android回调函数触发的几种方式 广播 静态对象

一.通过广播方式: 1.比如登录.假如下面这个方法是外界调用的,那么怎样在LoginActivity里面执行登录操作,成功之后在回调listener接口呢?如果是平常的类,可以通过构造函数将监听类对象传入即可.但是在Activity中不能传递监听对象,所以考虑使用广播来实现. public void login(final LoginOnClickListener listener) { Intent intent = new Intent(context, LoginActivity.clas

Android学习笔记(十三)——碎片(一)

碎片 碎片可看作另外一种形式的活动,可以创建碎片来包含视图. 碎片总是嵌入在活动中,一般有两种常见形式: 1.碎片A和碎片B分别处于不同的活动中,当选择碎片A中的某一项时,触发碎片B启动: 2.碎片A和碎片B处于同一个活动中,共享同一活动,以创建更佳的用户体验. 点此下载完整源码~(代码适用于本文章所讲) 1.创建一个名为"Fragments"的项目,在res/layout文件夹下,分别新建fragment1.xml.fragment2.xml:在当前包名下,分别新建Fragment1

【转】Pro Android学习笔记(十三):用户界面和控制(1):UI开发

目录(?)[-] UI开发 方式一通过XML文件 方式二通过代码 方式三XML代码 UI开发 先理清一些UI概念: view.widget.control:这三个名词其实没有什么区别,都是一个UI元素,例如一个button.Container(容器):包含其他view,例如grid可以认为是容器,它有cell,每个cell可以是一个view.Layout: 对容器们和view们的排版,可以包含其他layout. Android提供常用的UI(例如文字框)以及适合移动设备的控制,基础类为andro