Broadcast Reciever的使用

Broadcast Reciever的使用

Demo地址:https://code.csdn.net/kernel/broadcastreceiver
*注意*:Demo使用了roboguice的框架,和Broadcast Reciever里的一个隐式intent不能缺少setFlags(Intent.FLAG
ACTIVITYNEWTASK)

首先:创建自己的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>