Android四大组件之Broadcast Receiver

一、Android广播机制

Android每个应用程序都可以只对自己关心的广播进行注册,这些广播可以来自系统,也可以是其他应用程序,而接受广播的方法则需要借助广播接收器(Broadcast Reciever),Amdroid中的广播主要有两种,标准广播和有序广播。

1、标准广播

标准广播(Normal broadcast)是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何     先后顺序可言。这种广播效率比较高,但同事也意味着它是无法被截断的。工作流程图如:

2、有序广播

有序广播(Ordered Broadcast)是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能收到这台哦广播消息,当这个广播接受器中的逻辑执行完后,广播才会继续传递。所以此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先手到广播信息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播消息。工作流程图如:

、注册广播

广播接收器对自己感兴趣的广播进行注册,这样当相应的广播发出时,广播接收器就能收到该广播,注册的方式分为动态注册和静态注册,前者是在代码中注册,后者是在AndroidManifest.xml中注册。

1、动态注册

我们以监听网络变化为例,动态注册一个广播接收器。代码如下:

public class MainActivity extends Activity {
	private IntentFilter intentFilter;
	private NetworkChangeReceiver networkreceiver;
	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		intentFilter= new IntentFilter();
		intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
		networkreceiver = new NetworkChangeReceiver();
		registerReceiver(networkreceiver, intentFilter);
	}
	
	@Override
	protected void onPause() {
		// TODO Auto-generated method stub
		super.onPause();
		unregisterReceiver(networkreceiver);
	}
	
	class NetworkChangeReceiver extends BroadcastReceiver{

		@Override
		public void onReceive(Context context, Intent intent) {
			// TODO Auto-generated method stub
			Toast.makeText(context, "network changes", Toast.LENGTH_SHORT).show();
		}
		
	}
}

可以看到,在MainActivity中定义一个内部类NetworkChangeReceiver继承BroadcastReceiver,并重写onReceiver()方法。这样只要有广播来时,onReceiver()就会得到执行,具体的逻辑就在这个方法中处理。在onCreate()方法中,我们创建建了一个IntentFilter实例,并添加一个action,其值为android.net.conn.CONNECTIVITY_CHANGE,因为当网络状态发生改变时,系统会发出一条android.net.conn.CONNECTIVITY_CHANGE的广播,

也就是说我们想监听什么广播,添加相应的action就可以了。最后调用registerReceiver()方法进行注册,

将NetworkChangeReceiver实例和IntentFilter实例传进去。这样就可以实现网络状态的功能监听。

注意动态注册的的广播接收器一定要取消注册,我们是在onDestroy()方法中通过unregisterReceiver()方法来取消注册广播接收器。

2、静态注册

动态注册可以自由地控制注册与注销,但缺点就是必须在程序启动才能收到收到广播,而静态注册的方式可以让程序在为未启动的情况下 就能就收到广播。

下面通过静态注册广播接受器实现开机启动的功能,新建一个类BootCompleteReceiver继承BroadcastReceiver

public class BootCompleteReceiver extends BroadcastReceiver{

	@Override
	public void onReceive(Context context, Intent intent) {
		// TODO Auto-generated method stub
		Toast.makeText(context, "Boot Complete", Toast.LENGTH_SHORT).show();
	}
	
}
然后我们要在AndroidManifest.xml中将这个广播接收器的类名注册进去

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.gomez.broadcasttest"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="19"
        android:targetSdkVersion="22" />
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <receiver 
            android:name="com.gomez.broadcasttest.BootCompleteReceiver">
            <intent-filter >
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
            </intent-filter>
        </receiver>
    </application>

</manifest

在<application>标签内添加<receiver>标签,用法跟<activity>标签相似,通过android:name来指定具体注册哪一个广播接收器,

然后在<intent-filter>标签里加入想要接收的广播就行了,

由于Android系统启动完成后会发出android.intent.action.BOOT_COMPLETED的广播,因此我们在这里添加相应的action。

需要注意的是不要在onReceive()方法中添加过多的逻辑或者进行任何的耗时操作,因为在广播接受器中是不允许开启线程的,

onReceive()方法运行了较长时间而没有结束时,程序就会报错。

三、发送自定义广播

1、发送标准广播

button.setOnClickListener(new OnClickListener() {
			
			@Override
			public void onClick(View v) {
				Intent intent = new Intent("com.gomez.broadcasttest.MY_BROADCAST");
				sendBroadcast(intent);
			}
		});
首先创建一个Intent对象,传入com.gomez.broadcasttest.MY_BROADCAST这个值,然后调用sendBroadcast()方法传入Intent实例,把广播发送出去。

2、发送有序广播

button.setOnClickListener(new OnClickListener() {
			
			@Override
			public void onClick(View v) {
				Intent intent = new Intent("com.gomez.broadcasttest.MY_BROADCAST");
				sendOrderedBroadcast(intent, null);
			}
		});
区别在于调用sendOrderedBroadcast()方法,第一个参数传入Intent对象,第二个参数是接收器权限的参数,这里传入null就可以了。

如果想要在接收器中截断广播,可以调用abortBroadcast()方法来截断广播,这样后面的广播接收器将无法接收到这条广播。如下:

public class BootCompleteReceiver extends BroadcastReceiver{

	@Override
	public void onReceive(Context context, Intent intent) {
		// TODO Auto-generated method stub
		Toast.makeText(context, "Boot Complete", Toast.LENGTH_SHORT).show();
		abortBroadcast();
	}
	
}

四、本地广播

本地广播言下之意就是只能在本地传播,这样的发出的广播只能在应用程序的内部进行传递,用法如下:

public class LocalBroadcastActivity extends Activity{
	private IntentFilter intentFilter;
	private LocalRecevier localReceiver;
	private LocalBroadcastManager localBroadcastManager;
	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		// TODO Auto-generated method stub
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		//获取实例
		localBroadcastManager = LocalBroadcastManager.getInstance(this);
		//注册本地广播监听器
		intentFilter = new IntentFilter();
		intentFilter.addAction("com.gomez.broadcasttest.LOCAL_BROADCAST");
		localReceiver = new LocalRecevier();
		localBroadcastManager.registerReceiver(localReceiver, intentFilter);
		Button button = (Button) findViewById(R.id.button);
		button.setOnClickListener(new OnClickListener() {
			
			@Override
			public void onClick(View v) {
				// TODO Auto-generated method stub
				Intent intent = new Intent("com.gomez.broadcasttest.LOCAL_BROADCAST");
				localBroadcastManager.sendBroadcast(intent);//发送本地广播
			}
		});
		
	}
	@Override
	protected void onDestroy() {
		// TODO Auto-generated method stub
		super.onDestroy();
		localBroadcastManager.unregisterReceiver(localReceiver);
	}
	
	class LocalRecevier extends BroadcastReceiver{

		@Override
		public void onReceive(Context context, Intent intent) {
			Toast.makeText(context, "receive local broadcast", Toast.LENGTH_SHORT).show();
		}
	}
}












  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值