什么是廣播
安卓中的Broadcast,與傳統(tǒng)意義的電臺廣播類似,一個廣播可以有任意個接收者,接受者根據(jù)收到的廣播來做出相應(yīng)的動作。廣播是一個典型的觀察者模式,它的最大特點就是發(fā)送方完全不關(guān)心接收方是否接收到數(shù)據(jù),也不關(guān)心接收方如何處理數(shù)據(jù),同樣,接受者也可以監(jiān)聽多個廣播,發(fā)送和接收方是獨立運行的。
廣播的注冊
定義Receiver:
package com.example.broadcasttest;
public class BootCompReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context,Intent intent){
Toast.makeText(context, "系統(tǒng)啟動完畢", Toast.LENGTH_SHORT).show();
}
}
1.靜態(tài)廣播:在manifest文件中注冊
<receiver android:name="com.example.broadcasttest.BootCompReceiver">
<此處設(shè)置需要接收的廣播:示例是系統(tǒng)啟動之后發(fā)出的開機廣播和網(wǎng)絡(luò)狀態(tài)變化時的廣播/>
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
</receiver>
2.動態(tài)廣播:在代碼中采用動態(tài)注冊方式
public class MainActivity extends Activity {
public static final String BROADCAST_ACTION = "com.example.corn";
private BroadcastReceiver mBroadcastReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//此處注冊Receiver
mBroadcastReceiver = new BootCompReceiver ();
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("android.intent.action.BOOT_COMPLETED");
registerReceiver(mBroadcastReceiver, intentFilter);
}
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(mBroadcastReceiver);
}
}
需要注意的是,此處接收的為系統(tǒng)廣播,需要先聲明權(quán)限
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>"
兩種注冊方式的區(qū)別:
靜態(tài)注冊:即使APP已經(jīng)退出,靜態(tài)注冊的廣播接收器依然可以接受到廣播。但是從3.1之后,有所變化
動態(tài)注冊:當此Activity實例化時,會動態(tài)將MyBroadcastReceiver注冊到系統(tǒng)中。當此Activity銷毀時,動態(tài)注冊的MyBroadcastReceiver將不再接收
廣播的發(fā)送
1.標準廣播
接收器的執(zhí)行順序不確定,廣播發(fā)出之后,所有監(jiān)聽該廣播的接收器都可以接收到。
發(fā)送標準廣播:
package com.example.broadcasttest;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.layout_main);
final Intent intent = new Intent("com.example.broadcasttest.STANDARD_BROADCAST");
intent.putExtra("msg1", " 這是標準廣播");
Button sendButton = (Button) findViewById(R.id.send_broadcasr);
sendButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
sendBroadcast(intent);
}
});
這樣一個標準廣播就發(fā)送完成了。Receiver在注冊的時候,Action設(shè)置為"com.example.broadcasttest.STANDARD_BROADCAST",就可以接受到該廣播,同時可以獲取intent包含的信息
2.有序廣播
與標準廣播類似,不同的是,Receiver有一個priority參數(shù),可以定義權(quán)限,數(shù)值越大權(quán)限越高,接收時按照權(quán)限由高向低依次接收,先接收到的Receiver可以對廣播進行攔截或者加工intent信息,然后再向下傳遞。
有序廣播特有的方法:
abortBroadcast();
setResult: setResult(int code, String data, Bundle extras); setResultCode(int code);setResultData(String data);setResultExtras(Bundle extras);
getResult:getResultCode();getResultData(); getResultExtras(boolean makeMap);
示例:
發(fā)送廣播:
package com.example.broadcasttest;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.layout_main);
final Intent intent = new Intent("com.example.broadcasttest.ORDERED_BROADCAST");
intent.putExtra("msg1", " 這是有序廣播");
Button sendButton = (Button) findViewById(R.id.send_broadcasr);
sendButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
sendOrderedBroadcast(intent,null);//參數(shù)2可選,用來設(shè)置權(quán)限
}
});
與標準廣播基本相同,發(fā)送用sendOrderedBroadcast來代替
Receiver1:
public class FirstOrderedReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context,Intent intent){
String msg = intent.getStringExtra("msg");
CharSequence showmsg = "收到廣播:"+msg;
Toast.makeText(context, showmsg, Toast.LENGTH_SHORT).show();
setResultData(",第一個Receiver已處理完畢");
//abortBroadcast();
}
}
Receiver2:
public class SecondOrderedReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context,Intent intent){
String msg = intent.getStringExtra("msg");
CharSequence showmsg = "收到廣播:"+msg;
String data=getResultData();
Toast.makeText(context, "Receiver2:"+showmsg+"\n"+data, Toast.LENGTH_SHORT).show();
abortBroadcast();
}
}
注冊:
<receiver android:name="com.example.broadcasttest.FirstOrderedReceiver">
<intent-filter android:priority="100">
<action android:name="com.example.broadcasttest.ORDERED_BROADCAST"/>
</intent-filter>
</receiver>
<receiver android:name="com.example.broadcasttest.SecondOrderedReceiver">
<intent-filter android:priority="10">
<action android:name="com.example.broadcasttest.ORDERED_BROADCAST"/>
</intent-filter>
</receiver>
注意android:priority屬性,一個是100,一個是10,所以Receiver1先接收到廣播。
執(zhí)行結(jié)果:
收到廣播:這是有序廣播
Receiver2:收到廣播
第一個Receiver已處理完畢
Receiver2 Toast之后有abort語句,所以,如果還有其他的Receiver,將不會接收到該廣播。
廣播的作用范圍
以上所介紹的廣播,發(fā)出之后可以被系統(tǒng)中的任意應(yīng)用所接收,可以稱為全局廣播。在API21之后,新加了一個LocalBroadcastManager類,發(fā)出的廣播僅僅在應(yīng)用內(nèi)部被接收,可以稱為本地廣播。
示例:
定義Receiver:
class LocalReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent){
Toast.makeText(context, "Received Local Broadcast", Toast.LENGTH_SHORT).show();
}
}
發(fā)送并注冊本地廣播:
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);//獲取實例
Button localButton = (Button) findViewById(R.id.local_broadcast);
localButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Intent intent = new Intent("Local_Broadcast");
localBroadcastManager.sendBroadcast(intent);
}
});
//注冊本地廣播監(jiān)聽
intentFilter.addAction("Local_Broadcast");
localReceiver = new LocalReceiver();
localBroadcastManager.registerReceiver(localReceiver, intentFilter);
public void onDestroy(){//在onDestroy()方法中取消注冊。
super.onDestroy();
localBroadcastManager.unregisterReceiver(localReceiver);
}
本地廣播必須在程序啟動以后才生效,所以是不能采用靜態(tài)注冊的。
其他
在Receiver注冊的時候,有幾個屬性是需要注意的:
android:exported
該Receiver是否能接收其他app發(fā)出的廣播。如果Receiver有設(shè)置intent-filter,默認值為true,沒有intent-filter,默認為false。同樣,Activity/Service的該項默認值也是如此。
android:permission
如果設(shè)置了這項屬性,必須具有相應(yīng)權(quán)限的廣播發(fā)送方發(fā)送的廣播才能被接收
靜態(tài)廣播接收
Android 3.1開始系統(tǒng)在Intent與廣播相關(guān)的flag增加了參數(shù)
FLAG_INCLUDE_STOPPED_PACKAGES:包含已經(jīng)停止的包(停止:即包所在的進程已經(jīng)退出)
FLAG_EXCLUDE_STOPPED_PACKAGES:不包含已經(jīng)停止的包
自Android3.1開始,系統(tǒng)本身則增加了對所有app當前是否處于運行狀態(tài)的跟蹤。在發(fā)送廣播時,不管是什么廣播類型,系統(tǒng)默認直接增加了值為FLAG_EXCLUDE_STOPPED_PACKAGES的flag,導(dǎo)致即使是靜態(tài)注冊的廣播接收器,對于其所在進程已經(jīng)退出的app,同樣無法接收到廣播。
自己發(fā)出的廣播可以通過更改flag來實現(xiàn)App退出后也能接收廣播的功能,系統(tǒng)廣播的flag是無法更改的。解決方案是將service與APP設(shè)為不同進程。
onReceive的Context參數(shù)
對于靜態(tài)注冊廣播:context是ReceiverRestrictedContext
動態(tài)注冊的廣播:Context是注冊廣播的Activity
本地廣播:context是application
Sticky廣播
在API21已棄用,就不再介紹了。