浅谈Android Broadcast

前言:本文所写的是博主的个人见解,如有错误或者不恰当之处,欢迎私信博主,加以改正! 原文链接demo链接

广播简述

Android应用程序可以发送或者接收来自 Android 系统和其他 Android 应用程序的广播消息,类似于发布订阅设计模式。当感兴趣的事件发生时,这些广播被发送。例如,Android 系统在各种系统事件发生时发送广播,比如系统启动或者设备开始充电等。应用程序还可以发送自定义的广播,比如通知其他应用可能感兴趣的内容(例如,一些新数据已被下载)。

应用程序可以注册接收特定的广播,当发送广播时,系统自动将广播路由到订阅该特定类型广播的应用程序。一般来说,广播可以用作跨应用程序和正常用户流之外的消息传递系统。

系统广播

当发生各种系统事件时,系统会自动发送广播,例如系统切换飞行模式时,系统广播被发送到所有接收订阅事件的应用程序。

广播消息本身被包裹在一个 Intent 对象的动作字符串标识(例如 android.intent.action.AIRPLANE_MODE )。这个 Intent 还可以包括附加到其额外字段中的附加信息。比如,"飞行模式" 的 Intent 包括一个 boolean 额外指示是否 "飞行模式"。

系统广播行为的完整列表,请参阅 Android SDK 中的 BROADCAST_ACTIONS.TXT 文件。每个广播行为都有与之关联的常量字段,例如常量 ACTION_AIRPLANE_MODE_CHANGED 的值为 android.intent.action.AIRPLANE_MODE,每个广播动作的文档都可以在其相关联的常量字段中获得。

注:系统广播更改
Android 7 和更高的不再发送以下系统广播。这种优化影响所有的应用程序,不仅那些针对Android 7。

  • ACTION_NEW_PICTURE
  • ACTION_NEW_VIDEO
    针对 Android 7 的应用程序(API级别24)和更高的必须登记以下的广播代码注册广播接收器(BroadcastReceiver ,
    IntentFilter)。在清单中声明接收器不起作用。
  • CONNECTIVITY_ACTION

注册接收广播

应用程序可以通过两种方式接收广播:通过清单声明的接收者和上下文注册的接收者

  1. 清单声明的接收器(静态注册)

在清单中声明广播接收者,可以通过下面的步骤:
(1)在应用程序的清单中指定 <receiver> 元素

<receiver
       android:name=".DemoBroadcastReceiver"
       android:exported="true">
     <intent-filter>
         <action android:name="android.intent.action.BOOT_COMPLETED"/>
         <action android:name="android.intent.action.INPUT_METHOD_CHANGED"/>
     </intent-filter>
         
</receiver>

intent-filter (意图过滤器) 指定你的接收者订阅的广播操作。

(2)继承 BroadcastReceiver 并实现 onReceive(Context,Intent) 方法,请看下面的示例:

public class DemoBroadcastReceiver extends BroadcastReceiver {
     private static final String TAG = "DemoBroadcastReceiver";
     
     @Override
     public void onReceive(Context context, Intent intent) {
         StringBuilder sb = new StringBuilder();
         sb.append("Action: " + intent.getAction() + "\n");
         sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + "\n");
         String log = sb.toString();
         Log.d(TAG, log);
         Toast.makeText(context, log, Toast.LENGTH_LONG).show();
    }
    
}

当应用程序安装时,系统包管理器注册接受器,然后接收器将成为你的应用程序的单独入口,这意味着如果应用程序当前未运行,系统可以启动你的应用程序并发送广播。

系统将创建一个新的 BroadcastReceiver 组件对象来处理它接收的每个广播。这个对象仅对调用 onReceive(Context,Intent)的时候有效。当你在代码从该方法返回,系统将认为组件不再活动。

  1. context 注册接收者(动态注册)

    用context注册接收器,可以通过以下几个步骤:

    (1) 创建 BroadcastReceiver 并实例化

     BroadcastReceiver broadcastReceiver = new DemoBroadcastReceiver();
    

    (2) 创建一个 IntentFilter 并调用 registerReceiver( BroadcastReceiver , IntentFilter ) 来注册接收器

     IntentFilter intentFilter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION);
     intentFilter.addAction(Intent.ACTION_AIRPLANE_MODE_CHANGED);
     this.registerReceiver(broadcastReceiver,intentFilter);
    

    注意:要注册本地广播,请改用 LocalBroadcastManager.registerReceiver( BroadcastReceiver , IntentFilter )。

    如果context注册有效时,context注册的接收者就可以接收广播。例如你在 Activity 的 context 中注册,只要 Activity 没有被销毁,就会收到广播。如果是使用 ApplicationContext 进行注册,只要程序在运行就可以收到广播。

    (3)停止接收广播时,可以调用 unregisterReceiver( android.content.BroadcastReceiver ) 进行注销。当你不需要这个广播时或者 context 不再有效时,一定要注销接收器。·

    注意你在哪里注册和注销的接收者,比如,你在 Activity 的 onCreate( Bundle ) 中注册一个接收者,则应该在 onDestory() 中注销它,防止接收器泄露;如果你在 onResume() 中注册一个接收器,你应该在 onPause() 中注销,以防止多次注册( 如果你不想在 Activity 暂停时接收广播,可以做减少不必要的系统开销 )。不要在 onSaveInstanceState( Bundle ) 中注销,如果用户在历史栈中移回,则不会调用这个方法。

  2. 进程状态的影响

BroadcastReceiver 的状态(无论是否在运行)会影响其包含进程的状态,这也可能反过来影响其被系统杀死的可能性。例如,当进程执行一个接收器时(在其 onReceive() 方法中运行代码),它被认为是一个前台进程,除非内存压力极度大的时候,否则系统将保持其进程的运行。

然而,一旦你的代码从 onREceive() 方法中返回, BroadcastReceiver 就不在活动了。接收器的宿主进程变得和运行在这个进程中的其它应用程序组件一样重要。如果该进程只承载一个 manifest-declared 接收器(应用程序从来没有或者最近没有跟用户进行交互),然后从 onReceive() 返回时,系统将认为它的进程是一个低优先级的进程,很可能会杀死它,从而施放资源,用于其它优先级高的进程。

因此,不应该在广播接收器中执行耗时的后台线程,在执行 onReceive() 后,系统随时可以杀死进程以回收内存,这样做会终止在该进程中运行生成的线程。为了避免这种情况的发生,应该调用 goAsync() 方法(如果你需要更多的时间来处理后台线程中的广播)或使用 JobScheduler 从接收器调度 JobService ,则系统会知道该进程继续执行工作。

下面的这段代码显示了一个 BroadcastReceiver 使用 goAsync() 来标记,需要更多的时间来完成当前的操作,并在完成后再 onReceiver() 。如果你在 onReceive() 中需要的时间很长,导致 UI 线程错过了一帧( 16 ms ),可以使用上面的方法,在后台线程继续工作。

public class DemoBroadcastReceiver extends BroadcastReceiver {

    private static final String TAG = "DemoBroadcastReceiver";

    @Override
    public void onReceive(Context context, final Intent intent) {
        final PendingResult pendingResult = goAsync();

        AsyncTask<String, Integer, String> asyncTask = new AsyncTask<String, Integer, String>() {
            @Override
            protected String doInBackground(String... params) {
                StringBuilder sb = new StringBuilder();
                sb.append("Action: " + intent.getAction() + "\n");
                sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + "\n");
                String log = sb.toString();
                Log.d(TAG, log);

                // 必须调用 finish() ,以便 BroadcastReceiver 可以被回收。
                pendingResult.finish();

                return log;
            }
        };

        asyncTask.execute();
    }

}

发送广播

Android 提供了三种方法供应用发送广播:

  • sendOrderedBroadcast( Intent , String ) 方法将广播发送到一个接收器。随着每个接收器依次执行,它可以将结果传递到下一个接收器,也可以完全中止广播,使它不会传给其他接收器。Android 可以控制运行的命令接收器:匹配 intent-filter 的优先级属性;具备相同优先级的接收器,可以任意顺序运行

  • sendBroadcast( Intent ) 方法向所有接收器发送未确定顺序的广播,这被成为普通广播。这种广播更有效率,但是意味着接收器不能读取来自其他接收器的结果,传播从广播接收的数据,或者中止广播

  • LocalBroadcastManager.sendBroadcast 方法将广播发送到同个应用程序中的接收者。如果你不需要发送跨应用广播,请使用本地广播。实施 效率更高(无需进行跨进程通信),不需要担心其他可以接收或者发送广播的应用程序之间的任何安全问题

下面的代码片段演示了如何通过创建 Intent 并 调用sendBroadcast( Intent ) 来发送广播:

 Intent intent = new Intent();
 intent.setAction("com.passershowe.broadcast.NOTIFICATION_DEMO");
 intent.putExtra("data","Send a notice");
 sendBroadcast(intent);

广播消息被包装在 Intent 对象中,Intent 动作字符串必须提供应用程序的 java 包名和唯一的广播事件标识。你可以使用 putExtra( String , Bundle ) 附加信息到 Intent 中,还可以通过 Intent 调用 setPackage( String ) 将广播限制在同一组织中的一组应用程序。

注意:虽然 Intent 用于发送广播和使用 startActivity( Intent ) 启动 Activity ,但这些行为是完全无关的。广播接收器无法看到或者捕获用于开始 Activity 的 Intent ;同样,当你是广播 Intent 时,是无法找到或启动 Activity 。

广播权限约束

Permissions(权限) 允许你将广播限制为持有一定权限的应用程序集。可以对广播的发送方或接收方强制执行限制。

  1. 发送权限
    当你调用 sendBroadcast( Intent , String ) 或 sendOrderedBroadcast( Intent , String , BroadcastReceiver , Handle , int , String , Bundle ) 时,可以指定权限参数。只有通过其清单中的标签请求许可的接收器才能接收广播。例如,以下代码发送广播:
    sendBroadcast(new Intent("com.passershowe.broadcast.NOTIFY"),      Manifest.permission.SEND_SMS);
    

要接收广播,接收应用程序必须请求如下所示的权限:

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

你可以指定一个现有的系统权限(如SEND_SMS)或使用 <permission>元素定义自定义权限

注意:当你的应用程序注册了自定义权限,你必须在使用这个权限前,安装它

  1. 接收权限
    如果你在注册广播接收器(或者使用 registerReceiver( BroadcastReceiver , IntentFilter , String , Handle ) 或清单中的 <receiver> 标签)中指定权限参数,则只有使用 <uses-permission> 标签在其清单中(随后被授予许可是危险的)可以向接收者发送 Intent

    例如,假设你的接收应用程序具有清单声明的接收器,如下所示:

    <receiver android:name=".DemoBroadcastReceiver"
                   android:permission="android.permission.SEND_SMS">
            <intent-filter>
                <action android:name="android.intent.action.AIRPLANE_MODE"/>
            </intent-filter>
    </receiver>
    

    或者你的接收应用程序具有 context 注册的接收器,如下所示:

    IntentFilter filter = new IntentFilter(Intent.ACTION_AIRPLANE_MODE_CHANGED);
    registerReceiver(receiver, filter, Manifest.permission.SEND_SMS, null );
    

    然后,为了能够向这些接收者发送广播,发送应用程序必须如下所示请求许可:

    <uses-permission android:name="android.permission.SEND_SMS"/>
    
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容