BroadCast | Android 广播机制

123 阅读5分钟

Android 应用 <-> 系统/应用 , 类似于 [[发布-订阅]] 设计模式

这些广播会在所关注的事件发生时发送, Android 系统会在发生各种系统事件时发送广播。 应用自己也可以发送自定义广播来通知其他应用它们可能感兴趣的事件。

应用可以注册接收特定的广播,广播发出后,系统会自动将广播传送给同意接收这种广播的应用。

1.1 关于系统广播

==系统会在发生各种系统事件时自动发送广播==。广播消息本身会被封装在一个 Intent 对象中,该对象的操作字符串会标识所发生的事件(android.intent.action.AIRPLANE_MODE),也可以包含绑定到其他 extra 字段中的附加信息。

有关系统广播操作的完整列表,请参阅 Android SDK 中的 BROADCAST_ACTIONS.TXT 文件。

1.2 接收广播

  1. 清单声明的接收器
  2. 上下文注册的接收器

1. 清单声明的接收器

系统会在广播发出后启动你的应用(如果应用尚未运行)

清单中声明广播接收器的步骤:

  1. 在应用清单中指定 <receiver> 元素
    <receiver android:name=".MyBroadcastReceiver"  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 过滤器指定你的接收器所订阅的广播操作。

  1. 创建 BroadcastReceiver 子类并实现 onReceive(Context, Intent)

    public class MyBroadcastReceiver extends BroadcastReceiver {
            private static final String TAG = "MyBroadcastReceiver";
            @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();
            }
        }
    

系统软件包管理器(PMS) 会在应用安装时注册接收器, 之后该接收器会成为应用的一个独立入口点,一位置如果应用当前未运行,系统可以启动应用并发送广播。

系统会创建新的 BroadcastReceiver 组件对象来处理它接收到的每个广播。此对象仅在调用 onReceive(Context, Intent) 期间有效,一旦从此方法返回代码,系统便会认为该组件不再活跃。

2. 上下文注册的接收器

  1. 创建 BroadcastReceiver 的实例
BroadcastReceiver br = new MyBroadcastReceiver();
  1. 创建 IntentFilter 并调用 registerReceiver(BroadcastReceiver, IntentFilter) 来注册接收器:
IntentFilter filter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION);         filter.addAction(Intent.ACTION_AIRPLANE_MODE_CHANGED);          this.registerReceiver(br, filter);

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

只要注册上下文有效,上下文注册的接收器就会接收广播。

  1. 要停止接收广播,请调用 unregisterReceiver(android.content.BroadcastReceiver)。当您不再需要接收器或上下文不再有效时,请务必注销接收器。

请注意注册和注销接收器的位置,比方说,如果您使用 Activity 上下文在 onCreate(Bundle) 中注册接收器,则应在 onDestroy() 中注销,以防接收器从 Activity 上下文中泄露出去。如果您在 onResume() 中注册接收器,则应在 onPause() 中注销,以防多次注册接收器(如果您不想在暂停时接收广播,这样可以减少不必要的系统开销)。请勿在 onSaveInstanceState(Bundle) 中注销,因为如果用户在历史记录堆栈中后退,则不会调用此方法。

3. 对进程状态的影响

使用 goAsync( )来标记它在 onReceive 完成后需要更多时间才能完成。适用于后台线程。

 public class MyBroadcastReceiver extends BroadcastReceiver {
        private static final String TAG = "MyBroadcastReceiver";

        @Override
        public void onReceive(Context context, Intent intent) {
            final PendingResult pendingResult = goAsync();
            Task asyncTask = new Task(pendingResult, intent);
            asyncTask.execute();
        }

        private static class Task extends AsyncTask<String, Integer, String> {

            private final PendingResult pendingResult;
            private final Intent intent;

            private Task(PendingResult pendingResult, Intent intent) {
                this.pendingResult = pendingResult;
                this.intent = intent;
            }

            @Override
            protected String doInBackground(String... strings) {
                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);
                return log;
            }

            @Override
            protected void onPostExecute(String s) {
                super.onPostExecute(s);
                // Must call finish() so the BroadcastReceiver can be recycled.
                pendingResult.finish();
            }
        }
    }

1.3 发送广播

Android 为应用提供三种方式来发送广播:

  1. ==sendOrderedBroadcast(Intent, String)== 一次向一个接收器发送广播,顺序执行。
  2. ==sendBroadcast(Intent)== 按随机的顺序向所有接收器发送广播,常规广播。但也意味着接收器无法从其他接收器读取结果,无法传递从广播中收到的数据,也无法中止广播。
  3. ==LocalBroadcastManager.sendBroadcast== 将广播发送给与发送器位于同一应用中的接收器。如果不需要跨应用发送广播,请使用本地广播。

    Intent intent = new Intent();
    intent.setAction("com.example.broadcast.MY_NOTIFICATION");
    intent.putExtra("data","Notice me senpai!");
    sendBroadcast(intent);

广播消息封装在 Intent 对象中。

注意:虽然 intent 既用于发送广播,也用于通过 startActivity(Intent) 启动 Activity,但这两种操作是完全无关的。广播接收器无法查看或捕获用于启动 Activity 的 intent;同样,当您广播 intent 时,也无法找到或启动 Activity。

1.4 通过权限限制广播

可以通过权限将广播限定到拥有特定权限的一组应用。可以对广播的发送器或者接收器施加限制。

1. 带权限的发送


    sendBroadcast(new Intent("com.example.NOTIFY"),
                  Manifest.permission.SEND_SMS);
    

要接收此广播,接收方应用必须请求如下权限:


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

注意:自定义权限将在安装应用时注册。定义自定义权限的应用必须在使用自定义权限的应用之前安装。

2. 带权限的接收

如果在注册广播接收器时指定了权限参数,则广播方必须通过其清单中的 <uses-permission> 标记请求该权限(如果存在危险,则会被授予该权限),才能向该接收器发送 Intent。`

registerReceiver(BroadcastReceiver, IntentFilter, String, Handler) 

或清单中的 <receiver>


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

或者接收方应用具有如下所示的上下文注册的接收器:

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

那么发送方应用必须请求如下权限,才能向这些接收器发送广播:

一些调试方法

adb 命令查看广播队列:

logcat -s "BroadcastQueue"

dump 命令查看广播的记录:

dumpsys | grep BroadcastRecord

参考链接

developer.android.com/guide/compo…