看到Android中有关Broadcast Receiver组件方面的东西,广播被分为两种不同的类型:“普通广播(Normal broadcasts)”和“有序广播(Ordered broadcasts)”。普通广播是完全异步的,可以在同一时刻(逻辑上)被所有接收者接收到,消息传递的效率比较高,但缺点是:接收者不能将处理结果传递给下一个接收者,并且无法终止广播Intent的传播;然而有序广播是按照接收者声明的优先级别,被接收者依次接收广播。如:A的级别高于B,B的级别高于C,那么,广播先传给A,再传给B,***传给C 。优先级别声明在intent-filter元素的android:priority属性中,数越大优先级别越高,取值范围:-1000到1000,优先级别也可以调用IntentFilter对象的setPriority()进行设置 。有序广播的接收者可以终止广播Intent的传播,广播Intent的传播一旦终止,后面的接收者就无法接收到广播。另外,有序广播的接收者可以将数据传递给下一个接收者,如:A得到广播后,可以往它的结果对象中存入数据,当广播传给B时,B可以从A的结果对象中得到A存入的数据。
Context.sendBroadcast() 发送的是普通广播,所有订阅者都有机会获得并进行处理。
Context.sendOrderedBroadcast() 发送的是有序广播,系统会根据接收者声明的优先级别按顺序逐个执行接收者,前面的接收者有权终止广播(BroadcastReceiver.abortBroadcast()),如果广播被前面的接收者终止,后面的接收者就再也无法获取到广播。对于有序广播,前面的接收者可以将数据通过setResultExtras(Bundle)方法存放进结果对象,然后传给下一个接收者,下一个接收者通过代码:Bundle bundle = getResultExtras(true))可以获取上一个接收者存入在结果对象中的数据。
系统收到短信,发出的广播属于有序广播。如果想阻止用户收到短信,可以通过设置优先级,让你们自定义的接收者先获取到广播,然后终止广播,这样用户就接收不到短信了。
因此,可以用广播来实现一个黑名单的功能:
- package com.broadcastreceiver;
- import android.content.BroadcastReceiver;
- import android.content.Context;
- import android.content.Intent;
- import android.os.Bundle;
- import android.telephony.SmsMessage;
- import android.util.Log;
- public class FirstBroadCastReceiver extends BroadcastReceiver{
- @Override
- public void onReceive(Context context, Intent intent) {
- // TODO Auto-generated method stub
- if(intent.getAction().equals("android.provider.Telephony.SMS_RECEIVED")){
- Bundle bundle = intent.getExtras();
- StringBuilder phoneNum = new StringBuilder();
- //通过pdus可以获得接收到的所有信息
- Object[] objArray = (Object[]) bundle.get("pdus");
- //构建短信对象array,并依据收到的对象长度来创建array的大小
- SmsMessage[] messages = new SmsMessage[objArray.length];
- for (int i = 0; i < objArray.length; i++)
- {
- messages[i] = SmsMessage
- .createFromPdu((byte[]) objArray[i]);
- }
- //将送来的短信合并自定义信息于StringBuilder中
- for (SmsMessage currentMessage : messages)
- {
- //获得发送短信的电话号码
- phoneNum.append(currentMessage.getDisplayOriginatingAddress());
- }
- //这里可以添加数据库查询语句,用来查询发送短信的号码是否在黑名单中
- //注意,在onReceive()方法最长运行时间是10秒,如果超过10秒Android会认为该程序无响应
- //所以在BroadcastReceiver里不能做一些比较耗时的操作,否侧会弹出ANR(Application No Response)的对话框
- //如果需要完成一项比较耗时的工作,应该通过发送Intent给Service,由Service来完成
- //这里不能使用子线程来解决,因为 BroadcastReceiver的生命周期很短,子线程可能还没有结束BroadcastReceiver就先结束了
- //BroadcastReceiver一旦结束,此时BroadcastReceiver的所在进程很容易在系统需要内存时被优先杀死,
- //因为它属于空进程(没有任何活动组件的进程)。如果它的宿主进程被杀死,那么正在工作的子线程也会被杀死。
- //所以采用子线程来解决是不可靠的。
- if(phoneNum.toString().equals("254")){
- //如果在黑名单中,就终止广播,这样,系统就不会有任何提示,用户也就收不到短信了
- abortBroadcast();
- Log.e("msg", "sucess!!");
- }else
- Log.e("msg","fail!");
- }
- }
- }
既然要获取短信号码,就涉及到读短信的权限,同时intent-filter元素的android:priority属性表示优先级,越大说明优先级越高。
下面是AndroidManifest.xml代码:
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.broadcastreceiver"
- android:versionCode="1"
- android:versionName="1.0">
- <uses-permission android:name="android.permission.READ_SMS"></uses-permission>
- <uses-permission android:name="android.permission.RECEIVE_SMS"></uses-permission>
- <application android:icon="@drawable/icon" android:label="@string/app_name">
- <activity android:name=".MainBroadCastReceiver"
- android:label="@string/app_name">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <receiver android:name=".FirstBroadCastReceiver">
- <intent-filter android:priority="1000">
- <action android:name="android.provider.Telephony.SMS_RECEIVED"></action>
- </intent-filter>
- </receiver>
- </application>
- </manifest
【编辑推荐】