Broadcast Reciever的使用
Broadcast Reciever的使用
Demo地址:https://code.csdn.net/kernel/broadcastreceiver
*注意*:Demo使用了roboguice的框架,和Broadcast Reciever里的一个隐式intent不能缺少setFlags(Intent.FLAGACTIVITYNEWTASK)
首先:创建自己的BroadcastReceiver对象,需要继承android.content.BroadcastReceiver,并实现其onReceive方法。下面创建一个名为MyReceiver广播接收者:
MyReceiver.java广播接收者
package com.example.broadcast;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.widget.Toast;
/**
* @author 张鸿岩 422043466 2015年11月15日
*/
public class MyReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals("action_test")) {
String msg = intent.getStringExtra("msg");
Toast.makeText(context, msg, Toast.LENGTH_SHORT).show();
Intent in = new Intent();
Uri uri = Uri.parse("tel:10086");
in.setAction(Intent.ACTION_CALL);
in.setData(uri);
in.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);//其实flag和设置Activity的LaunchMode为SingleTask的作用是一样的
context.startActivity(in);
}
}
}
静态注册方法
静态注册是在AndroidManifest.xml文件中配置的,我们就来为MyReceiver注册一个广播地址:
<receiver android:name="com.example.broadcast.MyReceiver" >
<intent-filter>
<action android:name="action_test" />
</intent-filter>
</receiver>
动态注册方法
动态注册需要在代码中动态的指定广播地址并注册,通常我们是在Activity或Service注册一个广播,下面我们就来看一下注册的代码:
MyReceiver receiver = new MyReceiver();
IntentFilter filter = new IntentFilter();
filter.addAction("action_test");
registerReceiver(receiver, filter);
动态注销
注意,registerReceiver是android.content.ContextWrapper类中的方法,Activity和Service都继承了ContextWrapper,所以可以直接调用。在实际应用中,我们在Activity或Service中注册了一个BroadcastReceiver,当这个Activity或Service被销毁时如果没有解除注册,系统会报一个异常,提示我们是否忘记解除注册了。所以,记得在特定的地方执行解除注册操作:
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(receiver);
}
普通广播
普通广播对于多个接收者来说是完全异步的,通常每个接收者都无需等待即可以接收到广播,接收者相互之间不会有影响。对于这种广播,接收者无法终止广播,即无法阻止其他接收者的接收动作。 为了验证以上论断,新建三个BroadcastReceiver,演示一下这个过程,FirstReceiver、SecondReceiver和ThirdReceiver的代码如下:
FirstReceiver.java
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
public class FirstReceiver extends BroadcastReceiver {
private static final String TAG = "NormalBroadcast";
@Override
public void onReceive(Context context, Intent intent) {
String msg = intent.getStringExtra("msg");
Log.i(TAG, "FirstReceiver: " + msg);
}
}
SecondReceiver.java
package com.example.broadcast;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
/**
* @author 张鸿岩
*422043466
* 2015年11月15日
*/
public class SecondReceiver extends BroadcastReceiver {
private static final String TAG = "NormalBroadcast";
@Override
public void onReceive(Context context, Intent intent) {
String msg = intent.getStringExtra("msg");
Log.i(TAG, "SecondReceiver: " + msg);
}
}
ThirdReceiver.java
package com.example.broadcast;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
/**
* @author 张鸿岩
*422043466
* 2015年11月15日
*/
public class ThirdReceiver extends BroadcastReceiver {
private static final String TAG = "NormalBroadcast";
@Override
public void onReceive(Context context, Intent intent) {
String msg = intent.getStringExtra("msg");
Log.i(TAG, "ThirdReceiver: " + msg);
}
}
这三个接收者都接收到这条广播了,稍微修改一下三个接收者,在onReceive方法的最后一行添加以下代码,试图终止广播:
abortBroadcast();
再次点击发送按钮,控制台中三个接收者仍然都打印了自己的日志,表明接收者并不能终止广播。
有序广播
有序广播比较特殊,它每次只发送到优先级较高的接收者那里,然后由优先级高的接受者再传播到优先级低的接收者那里,优先级高的接收者有能力终止这个广播。 为了演示有序广播的流程,我们修改一下上面三个接收者的代码,如下:
FirstReceiver.java
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
public class FirstReceiver extends BroadcastReceiver {
private static final String TAG = "OrderedBroadcast";
@Override
public void onReceive(Context context, Intent intent) {
String msg = intent.getStringExtra("msg");
Log.i(TAG, "FirstReceiver: " + msg);
Bundle bundle = new Bundle();
bundle.putString("msg", msg + "@FirstReceiver");
setResultExtras(bundle);
}
}
在FirstReceiver和SecondReceiver中最后都使用了setResultExtras方法将一个Bundle对象设置为结果集对象,传递到下一个接收者那里,这样以来,优先级低的接收者可以用getResultExtras获取到最新的经过处理的信息集合。 代码改完之后,我们需要为三个接收者注册广播地址,我们修改一下AndroidMainfest.xml文件
<receiver android:name=".FirstReceiver">
<intent-filter android:priority="1000">
<action android:name="action_test"/>
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>
<receiver android:name=".SecondReceiver">
<intent-filter android:priority="999">
<action android:name="action_test"/>
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>
<receiver android:name=".ThirdReceiver">
<intent-filter android:priority="998">
<action android:name="action_test"/>
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>