美文网首页
Android四大组件学习之BroadcastReceiver

Android四大组件学习之BroadcastReceiver

作者: zoky_ | 来源:发表于2017-01-15 00:48 被阅读0次

    广播的相关概念

    BroadcastReceiver也就是“广播接收者”的意思,顾名思义,它就是用来接收来自系统和应用中的广播。
    在Android系统中,广播体现在方方面面,例如当开机完成后系统会产生一条广播,接收到这条广播就能实现开机启动服务的功能;当网络状态改变时系统会产生一条广播,接收到这条广播就能及时地做出提示和保存数据等操作;当电池电量改变时,系统会产生一条广播,接收到这条广播就能在电量低时告知用户及时保存进度等。
    虽然同属Android的四大组件,BroadcastReceiver也有自己独立的声明周期,但是和Activity、Service又不同。当在系统注册一个BroadcastReceiver之后,每次系统以一个Intent的形式发布Broadcast的时候,系统都会创建与之对应的BroadcastReceiver广播接收者实例,并自动触发它的onReceive()方法,当onReceive()方法被执行完成之后,BroadcastReceiver的实例就会被销毁。虽然它独自享用一个单独的进程,但也不是没有限制的,如果BroadcastReceiver.onReceive()方法不能在10秒内执行完成,Android系统就会认为该BroadcastReceiver对象无响应,然后弹出ANR(Application No Response)对话框,所以不要在BroadcastReceiver.onReceive()方法内执行一些耗时的操作。所以在BroadCast中不宜做一些耗时的操作。

    Broadcast的注册方法

    • 静态注册
      静态注册是在AndroidManifest.xml文件中配置的,我们就来为MyReceiver注册一个广播地址

      <receiver android:name=".MyReceiver">
      <intent-filter>
      <action android:name="android.intent.action.TEST_BROADCAST"/>
      <category android:name="android.intent.category.DEFAULT" />
      </intent-filter>
      </receiver>
      通过定义Receiver的action,只要是android.intent.action.XXX这个地址的广播,MyReceiver都能够接收得到,这种方式的注册是常驻型的,也就是说当应用关闭后,如果有广播信息传来,MyReceiver也会被系统调用而自动运行。

    • 动态注册
      动态注册需要在代码中动态的指定广播地址并注册,通常我们是在Activity或Service注册一个广播。

      MyReceiver receiver = new MyReceiver();

      IntentFilter filter = new IntentFilter();
      filter.addAction("android.intent.action.TEST_BROADCAST");

      registerReceiver(receiver, filter);

    registerReceiver是android.content.ContextWrapper类中的方法,Activity和Service都继承了ContextWrapper,所以可以直接调用。在实际应用中,我们在Activity或Service中注册了一个BroadcastReceiver,当这个Activity或Service被销毁时如果没有解除注册,系统会报一个异常,
    因此我们通常需要在onDestroy方法里释放广播。
    注册完广播接收器后,我们就可以发送一条广播了:

     Intent intent = new Intent("android.intent.action.TEST_BROADCAST");  
        intent.putExtra("msg", "hello receiver.");  
        sendBroadcast(intent);  
    }
    

    这样,action为TEST_BROADCAST的接收器都可以收到这条广播。

    广播的种类

    上面的例子只是一个接收者来接收广播,如果有多个接收者都注册了相同的广播地址,这就要看广播的种类来安排接受顺序了。

    • 普通广播(Normal Broadcast)
      Normal Broadcast,它是完全异步的,也就是说,在逻辑上,当一个Broadcast被发出之后,所有的与之匹配的BroadcastReceiver都同时接收到Broadcast。优点是传递效率比较高,但是也有缺点,就是一个BroadcastReceiver不能影响其他响应这条Broadcast的BroadcastReceiver。

    • 有序广播
      Ordered Broadcast,它是同步执行的,也就是说有序广播的接收器将会按照预先声明的优先级依次接受Broadcast,是链式结构,优先级越高(-1000~1000),越先被执行。因为是顺序执行,所有优先级高的接收器,可以把执行结果传入下一个接收器中,也可以终止Broadcast的传播(通过abortBroadcast()方法),一旦Broadcast的传播被终止,优先级低于它的接收器就不会再接收到这条Broadcast了。
      在接受有序广播时,BroadCastReceiver需要设置优先级, 即 android:priority="886" 属性,这个属性的范围在-1000到1000,数值越大,优先级越高。
      然后使用sendOrderedBroadcast方法来发送有序广播,这个方法需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为null,则表示接收者若要接收此广播,需声明指定权限。

    系统常见的广播:

    • android.intent.action.TIME_SET:系统时间被修改。
    • android.intent.action.DATE_CHANGED:系统日期被修改。
    • android.intent.action.BOOT_COMPLETED:系统启动完成。
    • android.intent.action.BATTERY_CHANGED:设备电量改变。
    • android.intent.action.BATTERY_LOW:设备电量低。
    • android.intent.action.ACTION_POWER_CONNECTED:设备连接电源。
    • android.intent.action.ACTION_POWER_DISCONNECTED:设备断开电源。
    • android.provider.Telephony.SMS_RECEIVED:系统收到短信。
    • android.intent.action.NEW_OUTGOING_CALL:拨打电话。

    广播使用示例

    从上面列举的一些动作会发布的Broadcast,可以找到,当系统接收到一条短信的时候,会发布一个“android.provider.Telephony.SMS_RECEIVED”的Broadcast,之前已经介绍过了,一般系统Broadcast都是有序广播,如果不被高优先级的BroadcastReceiver停止传递,会按照优先级顺序传递下去。

    而在这个示例中,通过监听接收短信的广播,当其内容有黑名单中的关键字的话,则阻止Broadcast继续传播,并使用Toast提示,否则正常提示短信信息。从上面列举的一些动作会发布的Broadcast,可以找到,当系统接收到一条短信的时候,会发布一个“android.provider.Telephony.SMS_RECEIVED”的Broadcast,一般系统Broadcast都是有序广播,如果不被高优先级的BroadcastReceiver停止传递,会按照优先级顺序传递下去。

    而在这个示例中,通过监听接收短信的广播,当其内容有黑名单中的关键字的话,则阻止Broadcast继续传播,并使用Toast提示,否则正常提示短信信息。

    public class MessageBroadcast extends BroadcastReceiver {
    
        private final String[] blackKeyWord = new String[] { "Message1", "Message2",
                "Message3" };
        @Override
        public void onReceive(Context context, Intent intent) {
            // 判断当前接收到的Broadcast是否是收到短信的action
            if (intent.getAction()
                    .equals("android.provider.Telephony.SMS_RECEIVED")) {
                StringBuilder sb = new StringBuilder();
                // 获取Broadcast传递的数据
                Bundle bundle = intent.getExtras();
                if (bundle != null) {
                    Object[] pdus = (Object[]) bundle.get("pdus");
                    for (Object p : pdus) {
                        byte[] pud = (byte[]) p;
                        // 声明一个SmsMessage,用于解析短信的byte[]数组
                        SmsMessage message = SmsMessage.createFromPdu(pud);
                        boolean flag = false;
                        for (String str : blackKeyWord) {
                            if (message.getMessageBody().contains(str) ) {
                                // 发现黑名单关键字,则标记为true
                                flag = true;
                                break;
                            }
                        }
                        if (flag) {
                            sb.append("发件人:\n");
                            sb.append(message.getOriginatingAddress());
                            sb.append("\n发送时间:\n");
                            Date date = new Date(message.getTimestampMillis());
                            SimpleDateFormat format = new SimpleDateFormat(
                                    "yyyy-MM-dd HH:mm:ss");
                            sb.append(format.format(date));
                            sb.append("\n短信内容:\n");
                            sb.append(message.getMessageBody());
                            Toast.makeText(context, sb.toString(),
                                    Toast.LENGTH_SHORT).show();
                        else{
                            // 如果存在黑名单关键字内容,停止Broadcast传播
                            abortBroadcast();
                        }
                            
                        }
                    }
                }
            }
        }
    }
    

    AndroidManifest.xml:

    <receiver android:name="cn.bgxt.Broadcastdemo.MessageWarn.MessageBroadcast">
    <!-- 设置优先级,短信优先级为0,大于0即可 -->
    <intent-filter android:priority="200">
    <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
    </intent-filter>
    </receiver>
    

    下一篇:
    Android四大组件学习之Service

    相关文章

      网友评论

          本文标题:Android四大组件学习之BroadcastReceiver

          本文链接:https://www.haomeiwen.com/subject/vgfybttx.html