天天看點

Android 基礎總結:(七)BroadcastReceiver詳解(上) Android 廣播的生命周期 Android 廣播應用

Android 廣播的生命周期

一個廣播接收者有一個回調方法:void onReceive(Context curContext, Intent broadcastMsg)。當一個廣播消息到達接收者時,Android調用它的onReceive()方法并傳遞給它包含消息的Intent對象。廣播接收者被認為僅當它執行這個方法時是活躍的。當onReceive()傳回後,它是不活躍的。

有一個活躍的廣播接收者的程序是受保護的,不會被殺死。但是系統可以在任何時候殺死僅有不活躍元件的程序,當占用的記憶體别的程序需要時。

這帶來一個問題,當一個廣播消息的響應時費時的,是以應該在獨立的線程中做這些事,遠離使用者界面其它元件運作的主線程。如果onReceive()衍生線程然後傳回,整個程序,包括新的線程,被判定為不活躍的(除非程序中的其它應用程式元件是活躍的),将使它處于被殺的危機。解決這個問題的方法是onReceive()啟動一個服務,及時服務做這個工作,是以系統知道程序中有活躍的工作在做。

Broadcast Receive為廣播接收器,它和事件處理機制類似,隻不過事件的處理機制是程式元件級别的,廣播處理機制是系統級别的。

Broadcast Receiver用于接收并處理廣播通知(broadcast announcements)。多數的廣播是系統發起的,如地域變換、電量不足、來電來信等。程式也可以播放一個廣播。程式可以有任意數量的 broadcast receivers來響應它覺得重要的通知。broadcast receiver可以通過多種方式通知使用者:啟動activity、使用NotificationManager、開啟背景燈、振動裝置、播放聲音等,最典型的是在狀态欄顯示一個圖示,這樣使用者就可以點它打開看通知内容。

通常我們的某個應用或系統本身在某些事件(電池電量不足、來電來短信)來臨時會廣播一個Intent出去,我們可以利用注冊一個Broadcast Receiver來監聽到這些Intent并擷取Intent中的資料。

Android 廣播應用

android中,不同程序之間傳遞資訊要用到廣播,可以有兩種方式來實作。

第一種方式:在Manifest.xml中注冊廣播,是一種比較推薦的方法,因為它不需要手動登出廣播(如果廣播未登出,程式退出時可能會出錯)。

具體實作在Manifest的application中添加:

<receiver android:name=".mEvtReceiver"> 
	<intent-filter>
		<action android:name="android.intent.action.BOOT_COMPLETED" />
	</intent-filter>
</receiver> 
           

上面兩個android:name分别是廣播名和廣播的動作(這裡的動作是表示系統啟動完成),如果要自己發送一個廣播,在代碼中為:

Intent i = new Intent("android.intent.action.BOOT_COMPLETED"); 
sendBroadcast(i);
           

這樣,廣播就發出去了,然後是接收。接收可以建立一個類,繼承至BroadcastReceiver,也可以建一個BroadcastReceiver的執行個體,然後得寫onReceive方法,實作如下:

protected BroadcastReceiver mEvtReceiver = new BroadcastReceiver() {
	@Override
	public void onReceive(Context context, Intent intent) {
		String action = intent.getAction();
		if (action.equals("android.intent.action.BOOT_COMPLETED")) {
			// Do something
		}
	}
}; 
           

完整執行個體:

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
	package="android.basic.lesson21"
	android:versionCode="1"
	android:versionName="1.0">
	<uses-sdk android:minSdkVersion="8" />
	<application android:icon="@drawable/icon" android:label="@string/app_name">
	<activity android:name=".MainBroadcastReceiver"
		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=".HelloBroadReciever">
		<intent-filter>
			<action android:name="android.basic.lesson21.Hello88" />
		</intent-filter>
	</receiver>
	</application>
</manifest> 
           
public class MainBroadcastReceiver extends Activity {
	/** Called when the activity is first created. */
	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		Button b1 = (Button) findViewById(R.id.Button01);
		b1.setOnClickListener(new View.OnClickListener() {
			@Override
			public void onClick(View v) {
				// 定義一個intent
				Intent intent = new Intent().setAction(
						"android.basic.lesson21.Hello88").putExtra("yaoyao",
						"yaoyao is 189 days old ,27 weeks -- 2010-08-10");
				System.out.println("sendBroadcast");
				// 廣播出去
				sendBroadcast(intent);
			}
		});
	}
}
           
public class HelloBroadReciever extends BroadcastReceiver {
	public HelloBroadReciever() {
		System.out.println("HelloBroadReciever");
	}
	// 如果接收的事件發生
	@Override
	public void onReceive(Context context, Intent intent) {
		// 對比Action決定輸出什麼資訊
		if (intent.getAction().equals("android.intent.action.BOOT_COMPLETED")) {
			Log.i("HelloBroadReciever",
					"BOOT_COMPLETED !!!!!!!!!!!!!!!!!!!!!!!!!");
		}
		if (intent.getAction().equals("android.basic.lesson21.Hello88")) {
			Log.i("HelloBroadReciever",
					"Say Hello to Yaoyao !!!!!!!!!!!!!!!!!!!!!!!!!");
			Log.i("HelloBroadReciever", intent.getStringExtra("yaoyao"));
		}
		Log.i("HelloBroadReciever", "onReceive**********");
		// 播放一首音樂
		// MediaPlayer.create(context, R.raw.babayetu).start();
	}
}
           

第二種方式,直接在代碼中實作,但需要手動注冊登出,實作如下:(以短信接收為例)

IntentFilter filter = new IntentFilter(); 
filter.addAction("android.provider.Telephony.SMS_RECEIVED"); 
           

registerReceiver(mEvtReceiver, filter); //這時注冊了一個recevier,名為mEvtReceiver,然後同樣用上面的方法以重寫onReceiver

最後在程式的onDestroy中要登出廣播,實作如下:

@Override
public void onDestroy() {
	super.onDestroy();
	unregisterReceiver(mPlayerEvtReceiver);
}
           

以短信接收為例:

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
	package="android.basic.lesson21"
	android:versionCode="1"
	android:versionName="1.0">
	<uses-sdk android:minSdkVersion="5" />
	<application android:icon="@drawable/icon" android:label="@string/app_name">
	<activity android:name=".MainBroadcastReceiver"
		android:label="@string/app_name">
		<intent-filter>
			<action android:name="android.intent.action.MAIN" />
			<category android:name="android.intent.category.LAUNCHER" />
		</intent-filter>
	</activity>
	</application>
	<uses-permission android:name="android.permission.RECEIVE_SMS"/>
</manifest>
           

增加一個短信權限<uses-permission android:name="android.permission.RECEIVE_SMS"/>

public class MainBroadcastReceiver extends Activity {
	/** Called when the activity is first created. */
	HelloBroadReciever br;
	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		Button b1 = (Button) findViewById(R.id.Button01);
		b1.setOnClickListener(new View.OnClickListener() {
			@Override
			public void onClick(View v) {
				Log.i("click", "OnClick");
				br = new HelloBroadReciever();
				IntentFilter filter = new IntentFilter();
				filter.addAction("android.provider.Telephony.SMS_RECEIVED");
				registerReceiver(br, filter);
			}
		});
	}
	@Override
	public void onDestroy() {
		super.onDestroy();
		unregisterReceiver(br);
		Log.i("br", "onDestroy");
	}
}
           
public class HelloBroadReciever extends BroadcastReceiver {
	public HelloBroadReciever() {
		System.out.println("HelloBroadReciever");
	}
	// 如果接收的事件發生
	@Override
	public void onReceive(Context context, Intent intent) {
		// 對比Action決定輸出什麼資訊
		if (intent.getAction()
				.equals("android.provider.Telephony.SMS_RECEIVED")) {
			Log.i("HelloBroadReciever",
					"SMS_RECEIVED !!!!!!!!!!!!!!!!!!!!!!!!!");
		}
		Log.i("HelloBroadReciever", "onReceive**********");
	}
}
           

先按開始鍵注冊廣播,然後在Emulator Control設定發短信如圖:

按Send,這時HelloBroadReciever的onReceive響應 


輸出SMS_RECEIVED !!!!!!!!!!!!!!!!!!!!!!!!! 

onReceive**********"資訊。


      

一個廣播接收者有一個回調方法:void onReceive(Context curContext, Intent broadcastMsg)。當一個廣播消息到達接收者是,Android調用它的onReceive()方法并傳遞給它包含消息的Intent對象。廣播接收者被認為僅當它執行這個方法時是活躍的。當onReceive()傳回後,它是不活躍的。

有一個活躍的廣播接收者的程序是受保護的,不會被殺死。但是系統可以在任何時候殺死僅有不活躍元件的程序,當占用的記憶體别的程序需要時。

這帶來一個問題,當一個廣播消息的響應時費時的,是以應該在獨立的線程中做這些事,遠離使用者界面其它元件運作的主線程。如果onReceive()衍生線程然後傳回,整個程序,包括新的線程,被判定為不活躍的(除非程序中的其它應用程式元件是活躍的),将使它處于被殺的危機。解決這個問題的方法是onReceive()啟動一個服務,及時服務做這個工作,是以系統知道程序中有活躍的工作在做。

繼續閱讀