BroadCast广播优先级问题

以前刚开始接触android时对诸如短信拦截之类的功能很感兴趣,网上很多文章都介绍了使用广播接收android.provider.Telephony.SMS_RECEIVED的动作,从而接收到SMS信息,不过当时也不知道怎么样把这个广播删掉,从而实现一个类似短信黑名单的功能。后来在网上看到可以使用abortBroadcast来屏蔽广播,我测试了一下,可行。

 

先上源码:

 

AndroidManifest.xml的代码:

 

<?xmlversion="1.0" encoding="utf-8"?>

 

<manifestxmlns:android="http://schemas.android.com/apk/res/android"

 

      package="com.study.sms"

 

     android:versionCode="1"

 

     android:versionName="1.0">

 

    <applicationandroid:icon="@drawable/icon" android:label="@string/app_name">

 

<receiverandroid:name=".smsReceiver">

 

<intent-filterandroid:priority="1000">

 

<actionandroid:name="android.provider.Telephony.SMS_RECEIVED"/>

 

</intent-filter>

 

</receiver>

 

    </application>

 

<uses-permissionandroid:name="android.permission.RECEIVE_SMS">

 

</uses-permission>

 

</manifest>

 

smsReceiver的代码:

 

packagecom.study.sms;

 

 

 

 

importandroid.content.BroadcastReceiver;

 

import android.content.Context;

 

import android.content.Intent;

 

import android.os.Bundle;

 

import android.telephony.SmsMessage;

 

 

 

 

publicclass smsReceiver extends BroadcastReceiver {

 

publicstatic final String SMS_RECEIVED_ACTION ="android.provider.Telephony.SMS_RECEIVED";

 

@Override

 

publicvoid onReceive(Context context, Intent intent) {

 

Stringaction = intent.getAction();

 

if(SMS_RECEIVED_ACTION.equals(action)){

 

Bundle bundle = intent.getExtras();

 

if (bundle != null){

 

Object[] pdus = (Object[])bundle.get("pdus");

 

for (Object pdu : pdus){

 

SmsMessage message = SmsMessage.createFromPdu((byte[]) pdu);

 

String sender = message.getOriginatingAddress();

 

if ("5556".equals(sender)){

 

//屏蔽手机号为5556的短信这里还可以时行一些处理如把这个信息发送到第三人的手机等等。

 

abortBroadcast();

 

}

 

}

 

}

 

}

 

}

 

 

 

 

}

 

测试的效果图:

 

 

 

使用5556给5554发送短信,5554无法收到。同时再起一个5558的avd给5554发送短信,一切正常。

 

之所以这个程序能实现屏蔽系统短信的功能,是因为广播有两种不同的类型:普通广播(normal broadcasts)和有序广播(ordered broadcasts)。普通广播是完全异步的,可以被所有的接收者接收到,并且接收者无法终止广播的传播。然而有序广播是按照接收者声明的优先级别,被接收者依次接收到。优先级别声明在 intent-filter 元素的 android:priority 属性中,数越大优先级别越高,取值范围:-1000到1000,优先级别也可以调用IntentFilter对象的setPriority()进行设置。有序广播的接收者可以终止广播Intent的传播,广播Intent的传播一旦终止,后面的接收者就无法接收到广播。上面的实例就是将smsReceiver的优先级设为1000。

 

 

 

Context.sendBroadcast()

 

发送的是普通广播,所有订阅者都有机会获得并进行处理。

 

Context.sendOrderedBroadcast()

 

发送的是有序广播,系统会根据接收者声明的优先级别按顺序逐个执行接收者。

 

 

 

我查看了一下系统发送android.provider.Telephony.SMS_RECEIVED广播的源码SMSDispatcher.java(frameworks\base\telephony\java\com\android\internal\telephony),证实其发送的是有序广播。

 

    protected void dispatchPdus(byte[][] pdus){

 

        Intent intent = newIntent(Intents.SMS_RECEIVED_ACTION);

 

        intent.putExtra("pdus",pdus);

 

        dispatch(intent,"android.permission.RECEIVE_SMS");

 

    }

 

   public void dispatch(Intent intent, Stringpermission) {

        // Hold a wake lock forWAKE_LOCK_TIMEOUT seconds, enough to give any

        // receivers time to take their ownwake locks.

        mWakeLock.acquire(WAKE_LOCK_TIMEOUT);

        mContext.sendOrderedBroadcast(intent,permission, mResultReceiver,

                this, Activity.RESULT_OK, null,null);

    }

 

 

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/prince58/archive/2011/03/10/6237792.aspx


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@SuppressLint("NewApi") public class MainActivity extends Activity { SmsReceiver myReceiver; Button btn; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); btn = (Button) findViewById(R.id.btn); btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { test(); } }); } public void test(){ Cursor cursor = null; String defaultSmsApp = Telephony.Sms.getDefaultSmsPackage(this); Intent intent = new Intent(Sms.Intents.ACTION_CHANGE_DEFAULT); intent.putExtra(Sms.Intents.EXTRA_PACKAGE_NAME, this.getPackageName()); startActivity(intent); try { cursor = getContentResolver().query(Uri.parse("content://sms/inbox"), new String[] { "_id", "address", "read" }, "read = ? ", new String[] {"0" }, "date desc"); if (cursor != null) { ContentValues values = new ContentValues(); values.put("read", "1"); for (cursor.moveToFirst(); !cursor.isAfterLast(); cursor.moveToNext()) { Log.v("cky", "" + cursor.getInt(cursor.getColumnIndex("_id")) + " , " + cursor.getString(cursor.getColumnIndex("address"))); int res = getContentResolver().update(Uri.parse("content://sms/inbox"), values, "_id=?", new String[] { "" + cursor.getInt(cursor.getColumnIndex("_id")) }); Log.i("cky","geng xin = "+res); } } intent = new Intent(Sms.Intents.ACTION_CHANGE_DEFAULT); intent.putExtra(Sms.Intents.EXTRA_PACKAGE_NAME, defaultSmsApp); startActivity(intent); } catch (Exception e) { e.printStackTrace(); } finally { if (cursor != null) { cursor.close(); cursor = null; } } } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值