前言
- 本篇主要介紹 廣播的類型、注冊和發送廣播、自定義廣播、本地廣播等
正文
一、廣播的類型
標準廣播,是一種完全異步執行的廣播,在廣播發出之后,幾乎所有的廣播接收器會在同一時刻接收到這條廣播,不分先后順序,優點是效率高,缺點是在廣播的過程中無法被截斷
有序廣播,是一種同步執行的廣播,在廣播發出之后,同一時刻只有一個廣播接收器能夠收到這條廣播消息,當這個廣播接收器中的邏輯侄子那個完畢之后,廣播才會繼續傳遞,根據優先級分先后順序,優先級高的可以截斷與其相鄰的優先級低的廣播
二、創建一個廣播接收者
//只需要寫一個類繼承自 BroadcastReceiver 即可,收到廣播之后會走 onReceive 方法
public class MyBroadcastReceriver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
//注意此方法中不能做耗時操作,其作用更多是為了打開某個組件
}
}
三、注冊廣播
第二點只是創建了一個廣播接收器,具體要接收什么樣的廣播主要是根據注冊廣播來決定的。注冊廣播分兩種方式,各有優缺點,分別為 動態注冊和靜態注冊
3.1 動態注冊,
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
MyBroadcastReceriver myBroadcastReceriver = new MyBroadcastReceriver();
IntentFilter filter = new IntentFilter();
filter.addAction("android.intent.action.NEW_OUTGOING_CALL");
registerReceiver(myBroadcastReceriver,filter);
如上所示,動態注冊了一個監聽外撥電話的廣播,但是要注意,動態注冊廣播一定要取消注冊,一般是在 onDestroy 方法中 調用 unregisterReceiver(myBroadcastReceriver); 來取消注冊
動態注冊可以自由地控制注冊和取消,比較靈活,但是有一個前提是程序運行之后才會開始接收廣播,如果說讓接收一條手機開機的廣播,動態注冊就不行了,這得需要靜態注冊。
3.2 靜態注冊
就拿注冊一個接收開機啟動的廣播為例
- 我們用 AndroidStudio 創建一個廣播之后,AndroidManifest.xml 中會添加一個節點,如下:
<receiver
android:name="broadcastreceiver.MyReceiver"
android:enabled="true"
android:exported="true">
</receiver>
- 我們需要在這個節點下添加一條 action 屬性,來表明接收的是什么樣的廣播,如下:
<receiver
android:name="broadcastreceiver.MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"></action>
</intent-filter>
</receiver>
系統開機完成之后,會發一條 android.intent.action.BOOT_COMPLETED 廣播,所以我們需要添加這個廣播
- 監聽手機開機 還需要添加一條權限
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
- 但是靜態注冊,不能取消
四、發送自定義廣播
之前我們接收的廣播都是系統定義好的,而實際項目中,我們很多情況都需要自定義廣播。
自定義廣播也是分兩個步驟,注冊廣播和發送廣播,注冊廣播也分兩種,靜態注冊和動態注冊,跟之前說的完全一樣,只不過是那個 action 由我們自己定義而已
4.1 發送標準廣播
就拿動態注冊來說吧
//注冊
MyBroadcastReceriver myBroadcastReceriver = new MyBroadcastReceriver();
IntentFilter filter = new IntentFilter();
filter.addAction("11111111");
registerReceiver(myBroadcastReceriver,filter);
//發送
Intent intent = new Intent();
intent.setAction("11111111");
sendBroadcast(intent);
我們創建一個廣播之后,在某個邏輯下 添加如上代碼,就表明注冊了一個 action 為 11111111 的廣播,當有人發送這條廣播的時候,我們的 MyBroadcastReceriver 就會收到該廣播。
4.2 發送有序廣播
- 相對于標準廣播來說,只需要在發送的時候呢改變一行代碼即可:
sendBroadcast(intent);
//改為
sendOrderedBroadcast(intent,null);
//第二個參數表示一個與權限相關的字符串,我們不需要,傳入 null 即可
4.3 設置廣播優先級,需要添加一條屬性,如下所示:
<receiver
android:name="broadcastreceiver.MyReceiver"
android:enabled="true"
android:exported="true">
<intent-filter
android:priority="1000"
>
<action android:name="android.intent.action.BOOT_COMPLETED"></action>
</intent-filter>
</receiver>
添加了一條 android:priority="1000" 屬性,其值的范圍在 -1000 和 1000 之間,數值越大,優先級越高
4.3 在有序廣播中截斷廣播
- 只需要在優先級高的廣播的 onReceive 方法中調用 abortBroadcast(); 就可截斷廣播
五、使用本地廣播
之前我們所說的所有廣播都是全局廣播,即發出的廣播可以被任何應用程序接收到,而我們的廣播接收器,也能接收到來自于其他任何應用程序的廣播,特別是我們在發送廣播的時候攜帶一些重要數據,這要是被其他人應用程序獲取到了,豈不是很不安全,而本地廣播就是為了解決這些問題
本地廣播主要是使用了一個 LocalBroadcastManager 類對廣播進行管理,如下所示:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);
//注冊廣播
MyBroadcastReceriver myBroadcastReceriver = new MyBroadcastReceriver();
IntentFilter filter = new IntentFilter();
filter.addAction("android.intent.action.NEW_OUTGOING_CALL");
localBroadcastManager.registerReceiver(myBroadcastReceriver,filter);
//發送廣播
Intent intent = new Intent();
intent.setAction("11111111");
localBroadcastManager.sendBroadcast(intent);
- 本地廣播除了之前的優點之外,還有一條就是比全局廣播更加高效