四大組件-----Broadcast詳解

什么是廣播

安卓中的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已棄用,就不再介紹了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容