Android四大組件——Broadcast Receiver

Broadcast Receiver

  完整代碼請見:longlong’s github

  • 廣播(Broadcas)是一種廣泛運用的在應用程序之間傳輸信息的機制 。而 廣播接收者是對發送出來的廣播進行過濾接收並響應的一類組件。廣播接收者(BroadcastReceiver)
  • BroadcastReceiver 自身並不實現圖形用戶界面,但是當它收到某個通知後, BroadcastReceiver 可以啓動Activity 作爲響應,或者通過 NotificationMananger 提醒用戶,或者啓動 Service 等等。
  • 標準廣播:完全異步執行的廣播 無法被截斷 所有的廣播接收器同步接收.
  • 有序廣播:同步執行的廣播 優先級高的廣播首先接收 可以截斷.

註冊廣播的兩種方式

動態註冊示例

  在你更改網絡狀態(數據連接)時會收到Toast提醒。

    public class MainActivity extends AppCompatActivity {
    private IntentFilter intentFilter;
    private NetworkChangeReceiver networkChangeReceiver;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        intentFilter = new IntentFilter();
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        networkChangeReceiver = new NetworkChangeReceiver();
        registerReceiver(networkChangeReceiver,intentFilter);

    }
    @Override
    protected void onDestroy(){
        super.onDestroy();
        unregisterReceiver(networkChangeReceiver);
    }
    class NetworkChangeReceiver extends BroadcastReceiver{

        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"network changes",Toast.LENGTH_LONG).show();
        }
    }
    }

  創建了一個IntentFilter實例,併爲它添加了一個值爲android.net.conn.CONNECTIVITY_CHANGE的action:當網絡狀態發生變化時,系統發出一條值爲此的廣播。另外,動態註冊的廣播接收器一定都要取消註冊。這裏是在onDestory()方法中調用unregisterReceiver()方法來實現的。
  更加人性化的代碼:告訴你網絡發生了什麼樣的變化,在onCreate()方法中,首先通過getSystemService()方法得到了ConnectivityManager()實例,這是一個系統服務類,專門用於管理網絡連接的。

    class NetworkChangeReceiver extends BroadcastReceiver{

        @Override
        public void onReceive(Context context, Intent intent) {
            ConnectivityManager connectivityManager =(ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
            NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
            if(networkInfo!=null&&networkInfo.isAvailable()){
                Toast.makeText(context,"network is available",Toast.LENGTH_LONG).show();
            }
            else {
                Toast.makeText(context,"network is unavailable",Toast.LENGTH_LONG).show();
            }
        }
    }

  另外,有一點很重要的說明:要在AndroidManifest.xml文件中加入權限

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>

靜態註冊示例

  實現開機啓動時的Toast提醒。
1. AndroidManifest.xml文件中:

  • 訪問權限:
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
  • 標籤<receiver>註冊:
    <receiver android:name=".MyBroadcastReceiver">
            <intent-filter android:priority="100">
                <action android:name="android.intent.action.BOOT_COMPLETED">
                </action>
            </intent-filter>
        </receiver>

2.直接新建一個BootBroadcastReceiver繼承自BroadcastReceiver,代碼如示:

    public class BootBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context,"Boot Complete",Toast.LENGTH_LONG).show();
    }

  不要在onReceive()方法中添加過多內容,因爲在廣播接收器中是不允許開線程的,當onReceive()方法運行了較長時間還沒有結束時,程序就會報錯。另外注意,BroadcastReceiver 會堵塞主線程。唯有 onReceive() 結束,主線程才得以繼續進行。

發送自定義廣播

  自定義廣播也要先定義一個廣播接收器 在AndroidManifest.xml文件對廣播接收器進行註冊。
  在按鈕的點擊事件中:

    public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button button  = (Button)findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent=new Intent("com.sunlianglong.broadcast.MY_BROADCAST");
                sendBroadcast(intent);
              //sendOrderedBroadcast(intent,null);
            }
        });
    }
    }

  這樣,所有監聽的廣播接收器就會收到Toast消息,此外,由於廣播是使用Intent進行傳遞的,因此你可以在Intent中攜帶一些數據。。。
  注:(1)發送有序廣播時,將sendBroadcast()方法改爲sendOrderedBroadcast()方法並傳入參數。
  (2)進行註冊時, <intent-filter android:priority="100">是設置優先級。
  (3)當A收到廣播後,可以向廣播中添加一些數據給下一個接收者(intent.putExtra()),或者終止廣播(在當前BroadcastReceiver內調用方法 abortBroadcast() )。

使用本地廣播

  (1)安全性提升,廣播只能在程序內部進行傳遞,廣播接收器也只能接受本應用程序發出的廣播。
  (2)只是使用了一個LocalBroadcastManager來對廣播進行管理,並提供了發送廣播和註冊廣播的方法:和前面所說的動態註冊廣播是一樣的。
  (3)本地廣播只能通過動態註冊來實現。因爲靜態註冊主要是爲了讓程序在未啓動的情況下也能收到廣播,發送本地廣播時程序完全已經啓動,也就不需要使用靜態註冊的功能。
  (4)如果你的廣播信息是用於應用的自我交流(不需要與其它應用協作),那麼建議使用LocalBroadcastManager.

需要收聽權限的廣播

  改變Activity中發送廣播的方法:

sendOrderedBroadcast(new Intent("com.sunlianglong.test.hahaha"), "com.sunlianglong.test");

  在發起廣播的應用中,需要在AndroidManifest文件中配置自定義的權限:

<permission android:protectionLevel="normal" android:name="com.sunlianglong.test"></permission>

  相應的,接收器所在的應用中必須設置接收權限:

uses-permission android:name="com.sunlianglong.test"></uses-permission>

注意

  1. 生命週期只有十秒左右,如果在 onReceive() 內做超過十秒內的事情,就會報ANR(Application No Response) 程序無響應的錯誤信息,如果需要完成一項比較耗時的工作 , 應該通過發送 Intent 給 Service, 由Service 來完成 . 這裏不能使用子線程來解決 , 因爲 BroadcastReceiver 的生命週期很短 , 子線程可能還沒有結束BroadcastReceiver 就先結束了 .BroadcastReceiver 一旦結束 , 此時 BroadcastReceiver 的所在進程很容易在系統需要內存時被優先殺死 , 因爲它屬於空進程 ( 沒有任何活動組件的進程 ). 如果它的宿主進程被殺死 , 那麼正在工作的子線程也會被殺死 . 所以採用子線程來解決是不可靠的。
  2. 動態註冊廣播接收器還有一個特點,就是當用來註冊的Activity關掉後,廣播也就失效了。靜態註冊無需擔憂廣播接收器是否被關閉,只要設備是開啓狀態,廣播接收器也是打開着的。也就是說哪怕app本身未啓動,該app訂閱的廣播在觸發時也會對它起作用。
  3. 系統常見廣播Intent,如開機啓動、電池電量變化、時間改變等廣播。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章