Android四大组件学习之BroadcastReceiver

广播的相关概念

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>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值