一、两种Broadcast:
· 普通广播(Normal Broadcast):用sendBroadcast()方法发送。
普通广播是完全异步的,逻辑上可以在同一时刻被所有匹配的接受者接收到,消息传递效率高,缺点是接受者不能将处理结果传递给下一个接收者,也无法终止广播传播。
ps:接收无序广播的接收器接收到广播的顺序也是有序的,接收无序广播的接收器也一样可以设置优先级的。如果同优先级的静态接收器想先接收某个广播,就要在包名上修改已让它在系统的报名清单中靠前显示。
· 有序广播(Ordered Broadcast):用sendOrderedBroadcast()方法发送。
有序广播的接收者们将按照事先生命的优先级依次接收,数越大优先级越高(取值范围:-1000~10000),优先级可以声明 在<intent-filter android:priority="n".../>,也可以调用IntentFilter对象的setPriority设置。并且接收者可以终止 传播(调用abortBroadcast()方法即可终止),一旦终止后面接收者就无法接受广播。另外,接受者可以将处理结果存入数据(可通过 setResultExtras(Bundle)方法将数据存入Broadcast),当做Broadcast再传递给下一级接收者(可通过代码 Bundle bundle = getResultExtras(true)获取上一级传递过来的数据)。
ps:有序广播在接受广播时的优先级遵循规则:
1动态注册>静态注册;
2在同为动态注册或者静态注册时,设置的优先级谁更大,谁的优先级高;
3当设置的优先级相同时,谁的app先安装,谁的优先级高;
4同app中设置的相同优先级时,那个广播写在清单文件的前面,谁的优先级高。
短信拦截原理:系统收到短信,发出的Broadcast属于有序广播,程序就可以通过设定优先级先接收到通知,然后终止传递。
ps:有些广播必须是动态注册的receiver才能收到
比如ACTION_SCREEN_ON,当屏幕被点亮的时候系统发送此广播,如果你尝试在manifest中注册receiver来接收,那么会失败, 这是为什么呢?他们在Intent中都设置了Intent.FLAG_RECEIVER_REGISTERED_ONLY,所以如果要接收,必须动态注册 广播接收器,ACTION_SCREEN_OFF也是如此。
二、举个例子(有序广播):
1、首先创建两个BroadcastReceiver。让第一个receive接收到广播后中断。
MyReceiver.java
1 public class MyReceiver extends BroadcastReceiver {
2
3 @Override
4 public void onReceive(Context context, Intent intent) {
5 System.out.println("MyReceiver接受到消息");
6 abortBroadcast(); //中断广播,不会再响比它有优先级低得广播再传播下去了
7 }
8 }
MyReceiver1.java
1 public class MyReceiver1 extends BroadcastReceiver {
2
3 @Override
4 public void onReceive(Context context, Intent intent) {
5 System.out.println("MyReceiver1接受到消息");
6 }
7 }
2、然后将两个receiver的action在AndroidManifest.xml文件中配置成一样的,并且设置成不同的优先级,代码如下:
1 <receiver android:name=".MyReceiver"> 2 <!-- priority优先级:数字越高优先级越高 --> 3 <intent-filter android:priority="5" > 4 <action android:name="com.codingblock.myreceiver.intent.action.MyReceiver"/> 5 </intent-filter> 6 </receiver> 7 <receiver android:name=".MyReceiver1"> 8 <intent-filter android:priority="4"> 9 <action android:name="com.codingblock.myreceiver.intent.action.MyReceiver"/> 10 </intent-filter> 11 </receiver>
3.最后在MainActivity中发送广播
1 public class MainActivity extends Activity {
2
3 Button btn_send_receiver;
4 @Override
5 protected void onCreate(Bundle savedInstanceState) {
6 super.onCreate(savedInstanceState);
7 setContentView(R.layout.activity_main);
8 btn_send_receiver = (Button)findViewById(R.id.btn_send_receiver);
9 btn_send_receiver.setOnClickListener(new OnClickListener() {
10
11 @Override
12 public void onClick(View v) {
13 Intent intent = new Intent();
14 intent.setAction("com.codingblock.myreceiver.intent.action.MyReceiver");
15 sendOrderedBroadcast(intent, null);//有序广播需要用sendOrderedBroadcast()方法发送
16 }
17 });
18 }
19 }
布局文件只有一个Button,比较简单,在此就不贴出了。
通过运行测试就会发现,点击发送消息按钮后只有MyReceiver接收到了消息,广播就被中断了。日志如下: