详解 Android 中的广播机制

移动开发 Android
Android 中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接受自己所关心的广播内容,这些广播内容可能是来自于系统,也可能是来自于其他应用程序的.

[[436469]]

前言

Android 中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接受自己所关心的广播内容,这些广播内容可能是来自于系统,也可能是来自于其他应用程序的;

Android 提供了一整套的 API,允许应用程序自由地发送和接受广播

今天我们就来详细介绍下

一. 广播介绍

1、标准广播

  • 这是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何先后顺序可言。接收器不能对收到的广播做任何处理,也不能截断广播继续传播;
  • 该种类的广播用sendBroadcast发送;

2、有序广播

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

3、粘性广播

  • 粘性广播的特点是Intent会一直保留到广播事件结束,而这种广播也没有所谓的10秒限制,10秒限制是指普通的广播如果onReceive方法执行时间太长,超过10秒的时候系统会将这个广播置为可以被干掉的‘候选人’,一旦系统资源不够的时候,就会干掉这个广播而让它不执行。该广播用sendStickyBroadcast发送;
  • 在Android5.0 & API 21中已经失效,所以不建议使用;

二、广播详解

Android 内置了很多系统级别的广播,我们可以在应用中通过监听这些广播来得到各种系统的状态信息。比如手机开机后会发送一条广播,电池的电量发生变化会发出一条广播,时间或时区发生改变也会发出一条广播等等;

注册广播方式一般有两种:

动态注册在代码中注册;

静态注册在 AndroidManifest.xml 中注册

1、静态注册

一般为常驻广播,在AndroidManifest.xml里通过标签声明

  1. <receiver android:name=".MyBroadcastReceiver"  android:exported="true"
  2.     <intent-filter> 
  3.         <action android:name="android.intent.action.BOOT_COMPLETED"/> 
  4.         <action android:name="android.intent.action.INPUT_METHOD_CHANGED" /> 
  5.     </intent-filter> 
  6. </receiver> 

intent过滤器里指定的是接收器订阅的action;

2、动态注册

非常驻广播,在使用时注册,用完及时销毁;

  1. BroadcastReceiver br = new MyBroadcastReceiver(); 
  2. IntentFilter filter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION); 
  3.     filter.addAction(Intent.ACTION_AIRPLANE_MODE_CHANGED); 
  4.     this.registerReceiver(br, filter); 

记得及时注销,以免内存泄漏;

  1. @Override 
  2.  protected void onDestroy() { 
  3.      super.onDestroy(); 
  4.      unregisterReceiver(receiver); 
  5.  } 

3、自定义广播

注册广播

  1.  <receiver 
  2.             android:name=".test.MyBroadcastReceiver" 
  3.             android:enabled="true" 
  4.             android:exported="true"
  5.             <intent-filter> 
  6.                 <action android:name="com.test.test" /> 
  7.             </intent-filter> 
  8.   </receiver> 
  9. public class MyBroadcastReceiver extends BroadcastReceiver { 
  10.     @Override 
  11.     public void onReceive(Context context, Intent intent) { 
  12.         Toast.makeText(context,"发送标准广播",Toast.LENGTH_LONG).show(); 
  13.     } 
  14.   Intent intent = new Intent("com.test.test"); 
  15.                  sendBroadcast(intent); 
  16. //发送有序广播 
  17.  Intent intent = new Intent("com.test.test"); 
  18.                  sendOrderedBroadcast(intent,null);  

4、本地广播

我们发送和接受的广播全部属于系统全局广播,即发出的广播可以被其它任何应用程序接收到,并且我们也可以接受来自于其它任何应用程序的广播;

为了解决广播安全性问题,Android 引入了一套本地广播机制,使用这个机制发出的广播只能在应用程序内部进行传递,并且广播接受器也只能接受来自本应用程序发出的广播,这样所有的安全性问题就都不存在了;

初始化广播:

  1. private LocalBroadcastManager localBroadcastManager; 
  2.    private void init() { 
  3.        //获取实例 
  4.        localBroadcastManager = LocalBroadcastManager.getInstance(this); 
  5.        IntentFilter intentFilter = new IntentFilter(); 
  6.        intentFilter.addAction("om.test.LOCAL_BROADCAST"); 
  7.        LocalReceiver localReceiver = new LocalReceiver(); 
  8.        localBroadcastManager.registerReceiver(localReceiver, intentFilter); 
  9.    } 
  10. Intent intent = new Intent("om.test.LOCAL_BROADCAST"); 
  11.        localBroadcastManager.sendBroadcast(intent); 
  12. 定义 
  13.    private class LocalReceiver extends BroadcastReceiver { 
  14.        @Override 
  15.        public void onReceive(Context context, Intent intent) { 
  16.            Toast.makeText(context, "本地广播.....", Toast.LENGTH_LONG).show(); 
  17.        } 
  18.    } 

优点:

  • 可以明确知道正在发送的广播不会离开我们的程序,因此不必担心机密数据泄露:
  • 其它的程序无法将广播发送到我们程序内部,因此不需要担心会有安全漏洞问题:
  • 发送本地广播比发送系统全局广播更加高效:

5、带权限的标准广播

发广播

当调用sendBroadcast(Intent, String)或sendOrderedBroadcast(Intent, String, BroadcastReceiver, Handler, int, String, Bundle)时,可以指定权限参数;

  1. sendBroadcast(new Intent("com.test"), 
  2.               Manifest.permission.SEND_SMS); 

要想接收这个广播,那么这个接收的app必须申请该权限

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

接收广播

如果在注册广播接收器时指定了权限参数(使用registerReceiver(BroadcastReceiver,IntentFilter,String,Handler)或清单文件中的里),则只有在清单文件中使用请求权限的广播发送者才可以将Intent发送给接收者;

在清单文件中声明:

  1. <receiver android:name=".MyBroadcastReceiver" 
  2.           android:permission="android.permission.SEND_SMS"
  3.     <intent-filter> 
  4.         <action android:name="com.test"/> 
  5.     </intent-filter> 
  6. </receiver> 

在注册的时候声明:

  1. IntentFilter filter = new IntentFilter("com.test); 
  2. registerReceiver(receiver, filter, Manifest.permission.SEND_SMS, null ); 

要给它发消息,那广播发送的app就必须得申请获得相应的权限才行:

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

总结

动态注册广播不是常驻型广播,也就是说广播跟随Activity的生命周期。注意在Activity结束前,移除广播接收器;

静态注册是常驻型,也就是说当应用程序关闭后,如果有信息广播来,程序也会被系统调用自动运行;

当广播为有序广播时:优先级高的先接收(不分静态和动态)。同优先级的广播接收器,动态优先于静态;

同优先级的同类广播接收器,静态:先扫描的优先于后扫描的,动态:先注册的优先于后注册的;

当广播为默认广播时:无视优先级,动态广播接收器优先于静态广播接收器。同优先级的同类广播接收器,静态:先扫描的优先于后扫描的,动态:先注册的优先于后册的;

本文转载自微信公众号「Android开发编程」

责任编辑:姜华 来源: Android开发编程
相关推荐

2013-03-26 13:55:45

Android Bro

2017-05-15 19:40:40

AndroidIPC机制

2013-03-28 09:07:37

Android开发Intent机制

2011-09-27 10:23:24

Java反射机制

2012-05-25 09:09:25

Windows Pho

2010-07-07 18:34:43

UML公共机制

2013-03-28 16:12:12

Message机制应用

2013-04-11 12:40:16

Android消息机制

2010-06-08 18:01:00

UML组成

2011-06-29 17:20:20

Qt 内存 QOBJECT

2011-05-26 15:23:34

JavaReflection

2009-09-23 16:30:01

Hibernate f

2009-04-10 09:55:44

C#反射.NET

2020-04-24 15:44:50

MySQL数据库锁机制

2009-07-15 09:15:26

2021-09-01 06:48:16

AndroidGlide缓存

2023-03-06 10:44:50

AndroidProguard

2021-03-02 09:12:25

Java异常机制

2010-04-29 15:50:35

2011-04-01 14:50:56

Java的反射机制
点赞
收藏

51CTO技术栈公众号