Android组件- 广播的静态动态有序无序与优先级

设计模式中有一种比较常用的模式——观察者模式:观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时接收某一个主题对象的消息。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己。Android中的广播接收器,就是观察者模式经典的案例,基于对观察者模式的理解,我们也应当对BroadcastReceiver的优缺点有个基本认识。

优点:广播发送者与接收者的耦合度很低,可以非常便利的解耦重构接收者的业务逻辑,解除耦合关系。

缺点:如果广播发送者有很多的接收者的话,要通知到所有的接收者需要一定的时间。


对BroadcastReceiver的设计思想有了一定了解之后,我们再来看Android BroadcastReceiver的一些关键知识点:

一. 普通(无序)广播和有序广播

1. 普通(无序)广播: 所有接收器都是异步处理的,所有注册的广播接收器在同一时间(理论上~)收到广播消息,收到的消息都是一致的,互相不能改变广播消息,也不能终止广播的发送。

2. 有序广播:做为对立面,可以想象普通广播不能做的,在这里可以做。

a). 广播消息是按照注册接收器时声明的优先级,按顺序发给各个接收器的,优先级最大可以声明为int型的最大值,也就是2147483647(为了最高优先级,拼了)。

b). 优先收到广播的接收者,可以改变消息的内容,将自己的处理结果发给下一个接收者。当然,也可以直接中断这个广播,后面的接收者就无法接收了。

有人就有江湖,有优先级就有竞争,一个典型的案例就是市场上各个通讯录产品,比如QQ,91通讯录,为什么他们可以拦截系统短信,把新的短信放到自己的库里面呢?其实原理很简单,就是有序广播。Android系统的短信是通过有序广播发送的,系统原生的短信app也是通过注册BroadcastReceiver来处理短信,因此我们只要在注册一个短信接收器,把优先级调到最最高的2147483647(QQ,91们估计就是这么干的),就能在系统原生的短信app之前,接收到短信广播,接下来爱干嘛干嘛了。

好奇的同学估计会问,都是2147483647,顺序是怎样的呢?答,谁先注册,谁先调用。

好奇的同学估计还会问,都是2147483647,有没可能都是我注册的先调用?答,有可能,通过动态注册!因为相同优先级的广播接收器,动态注册的会排在静态注册的前面。

接下来,就让我们来看第二个关键知识点。


二. 静态广播和动态广播

1. 先说明下,这里把静态注册的广播简称为静态广播,动态注册的广播简称为动态广播。

静态注册是通过AndroidManifest的标签<receiver>来实现的:

<receiver android:name="com.yourapp.demoReceiver">
            <intent-filter>
                <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
            </intent-filter>
</receiver>

动态广播则是通过代码来注册的:

public class BrocastReceiverDemo extends Activity {
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		IntentFilter filter = new IntentFilter();
		filter.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
		
		this.registerReceiver(connectChangedReceiver, filter);
	}
	
	@Override
	protected void onDestroy() {
		super.onDestroy();
		if (connectChangedReceiver != null)
			this.unregisterReceiver(connectChangedReceiver);
	}
	
	BroadcastReceiver connectChangedReceiver = new BroadcastReceiver() {
		@Override
		public void onReceive(Context context, Intent intent) {
			//doSomething();
		}
	};
}
2. 动态广播能做的,静态广播也能做,但是反过来就不成立了,接下来就一起看看静态广播有哪些特性。

a). 静态广播不依赖于应用的进程,换句话说就是即使我们的应用没开启,也可以接收广播。市场上一大票开机自启动的应用,就是注册了手机的开机广播来实现的。

b). 静态广播可以增加android:exported="false"的设置,限制只能接收本应用的广播消息。注意,加了android:exported="false",系统的广播也接收不到了

c). 关于a中说明的特性,在4.X系统中有一个bug,如果我们的app在安装过后从未启动,静态广播不会生效。虽然网上贴有一些特殊的方法来解决,但还是有ROM的适配问题(有兴趣的同学可以自行百度,关键字intent.setFlags(32))。


三. 广播接收器的其他要点

1. BroadcastReceiver.onReceive默认是在主线程(UI线程)执行的,这也是老生常谈的“千万不能在广播接收器做耗时操作,会ANR!”的真正原因

2. 广播发送者可以指定接收广播的应用,只要将sendBroadcast(intent)的intent加上具体的包名就行了,intent.setPackage("yourpackagename")。

3. 广播发送者和接收者都可以声明权限,这样只有在AndroidManifest声明对应权限,才能接收到相应的广播消息。



  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值