Android入门(11)| 全局广播与本地广播
文章目錄
- 廣播概念
- 接收廣播
- 動態(tài)注冊
- 實例
- 靜態(tài)注冊
- 實例
- 發(fā)送廣播
- 發(fā)送標(biāo)準(zhǔn)廣播
- 廣播的跨進程特性
- 發(fā)送有序廣播
- 本地廣播
廣播概念
Android 中的每個應(yīng)用程序都可以對自己感興趣的廣播進行注冊,這樣該程序就只會接收到自己所關(guān)心的廣播內(nèi)容,這些廣播可能是來自系統(tǒng)的,也可能是來自于其他應(yīng)用程序的。
廣播有兩種類型——有序廣播和標(biāo)準(zhǔn)廣播:
- 標(biāo)準(zhǔn)廣播: 一種完全異步執(zhí)行的廣播,在廣播發(fā)出去之后,所有的廣播接收器幾乎都會同一時刻接收到這條廣播消息,沒有任何的先后順序可言,這種廣播的效率比較高,但是無法被截斷。
- 有序廣播: 是一種同步執(zhí)行的廣播,在廣播發(fā)出去之后,同一時刻只會有一個廣播接收器能夠收到這條消息,當(dāng)這個廣播接收器中的邏輯執(zhí)行完畢之后,廣播才會繼續(xù)傳遞,所以這時候的廣播接收器是有優(yōu)先級順序的,并且前面的廣播接收器還可以截斷正在傳遞的廣播,這樣后面的廣播就無法收到廣播消息。
接收廣播
動態(tài)注冊
Android內(nèi)置了很多系統(tǒng)級別的廣播,我們可以在應(yīng)用程序中通過監(jiān)聽這些廣播來得到各種系統(tǒng)的狀態(tài)信息。比如手機開機、電池電量發(fā)生變化、時間或者時區(qū)發(fā)生改變等等。如果想要接收到這些廣播就需要使用廣播接收器。
注冊廣播的方式一般也有兩種,在 代碼中注冊(動態(tài)注冊) 或者 在 AndroidManifest.xml 中注冊(靜態(tài)注冊)。
實例
我們實現(xiàn)一個能準(zhǔn)確地告訴用戶當(dāng)前有沒有網(wǎng)絡(luò)的功能,在實現(xiàn)代碼前,由于 Android 為了保護用戶設(shè)備的隱私和安全,規(guī)定了程序需要進行一些對用戶來說比較敏感的操作,必須在配置文件中聲明權(quán)限才可以,該功能監(jiān)聽了網(wǎng)絡(luò)的變化,所以必須在 AndroidManifest.xml 配置權(quán)限才能訪問系統(tǒng)網(wǎng)絡(luò)狀態(tài):
靜態(tài)注冊
動態(tài)注冊的廣播可以自由控制注冊、注銷,很靈活,缺點是必須在程序啟動之后才能接受到廣播,因為注冊的邏輯是寫在 onCreate() 里面的,使用靜態(tài)注冊可以讓程序在未啟動的情況下接受到廣播。
實例
靜態(tài)注冊需要在 AndroidManifest.xml 進行注冊,使用 receiver 標(biāo)簽,并告訴這個 receiver 注冊哪一個 action,下面是一個開機啟動接受廣播的案例:
public class BootCompleteReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// TODO: This method is called when the BroadcastReceiver is receiving// an Intent broadcast.Toast.makeText(context, "Boot Complete", Toast.LENGTH_SHORT).show();} }PS:不要在 onReceive() 方法中添加過多邏輯或耗時操作,因為廣播接收器中不允許開啟線程,因此當(dāng) onReceive() 運行較長時間卻未結(jié)束時,程序就會報錯。
發(fā)送廣播
發(fā)送標(biāo)準(zhǔn)廣播
發(fā)送廣播使用 Intent 進行發(fā)送,首先需要準(zhǔn)備一個接收器用于接受發(fā)送的廣播:
public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 收到自定義廣播時會彈出提示Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_LONG).show();} }在 AndroidManifest.xml 中注冊廣播的值:
實現(xiàn)點擊 BroadcastActivity 活動中的 send broadcast 按鈕來發(fā)送廣播:
布局文件 broad_layout.xml:
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><Buttonandroid:id="@+id/button_broadcast1"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="send broadcast"/> </LinearLayout>活動代碼 BroadcastActivity.java:
public class BroadcastActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.broad_layout);Button button = findViewById(R.id.button_broadcast1);button.setOnClickListener((View v)->{// 將要發(fā)送的廣播植入IntentIntent intent = new Intent("com.example.activitytest.Activity.MY_BROADCAST");// 參數(shù)1:包名;參數(shù)2:接收器的路徑ComponentName componentName = new ComponentName("com.example.activitytest","com.example.activitytest.Activity.MyBroadcastReceiver");// 通過調(diào)用Intent中的setComponent方法,我們可以打開另外一個應(yīng)用中的Activity或者服務(wù)。intent.setComponent(componentName);// 調(diào)用Context的sendBroadcast方法發(fā)送廣播sendBroadcast(intent);});} }PS: ComponentName 構(gòu)造函數(shù)的第一個參數(shù)指的包名是 AndroidManifest.xml 文件下 package 屬性對應(yīng)的包名:
而非 BroadcastActivity.java 文件所在的包名:
詳見該博客
運行結(jié)果:
廣播的跨進程特性
新建一個項目,創(chuàng)建廣播接收器 MyReceiver.java :
public class MyReceiver extends BroadcastReceiver {private static final String TAG = "MyReceiver";@Overridepublic void onReceive(Context context, Intent intent) {Log.e(TAG, "onReceive: gone");Toast.makeText(context, "received gone", Toast.LENGTH_LONG).show();} }AndroidManifest.xml :
在原來項目的 BroadcastActivity.java 文件中發(fā)送第二條廣播:
運行結(jié)果:
更多關(guān)于廣播的問題詳見本文——解決 Android 8.0 以上靜態(tài)廣播無法注冊
發(fā)送有序廣播
很多人對之前的代碼可能會有疑問,我指定廣播發(fā)送給哪個包的哪個接收器,這還算“廣播”嗎?因此,對于安卓高版本而言,還有另一種發(fā)送廣播的方法:
修改 BroadcastActivity.java 中的代碼:
即可實現(xiàn)真正意義上的廣播。在此基礎(chǔ)上,我們發(fā)送有序廣播,定義接收器的優(yōu)先級:
并在接收器 MyBroadcastReceiver.java 中截斷廣播,不允許廣播繼續(xù)傳遞:
不必對另一個接收器 MyReceiver 進行更改,此時就已達到了只有 MyBroadcastReceiver 能收到廣播,而 MyReceiver 不能收到廣播的目的。
本地廣播
前面我們發(fā)送和接受的廣播都是系統(tǒng)的全局廣播,發(fā)出的廣播可以被其他任何應(yīng)用程序接收到。這樣容易引起安全問題,為了解決安全性問題,Android 支持發(fā)送本地廣播,其有以下特點:
- 廣播不會離開我們的程序,無需擔(dān)心機密數(shù)據(jù)泄露;
- 其他程序的廣播無法發(fā)送到我們程序內(nèi)部,無需擔(dān)心有安全漏洞的隱患;
- 比發(fā)送全局廣播更高效。
- 本地廣播的接收只能使用動態(tài)注冊,因為靜態(tài)注冊就是為了讓程序在未啟動的時候也能接收到廣播,而發(fā)送本地廣播的時候應(yīng)用程序肯定啟動了。
本地廣播并不復(fù)雜,主要就是使用了一個 LocalBroadcastManager 來對廣播進行管理,并且提供了發(fā)送廣播和注冊廣播接收器的方法:
public class BroadcastActivity extends AppCompatActivity {private static final String TAG = "BroadcastActivity";private IntentFilter intentFilter; // 意圖過濾器private LocalReceiver localReceiver; // 自定義接收器類private LocalBroadcastManager localBroadcastManager; // support包提供的本地廣播工具@SuppressLint("WrongConstant")@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.broad_layout);localBroadcastManager = LocalBroadcastManager.getInstance(this); // 獲取實例Button button = findViewById(R.id.button_broadcast1);button.setOnClickListener((View v)->{Log.e(TAG, "onCreate: start");// 將要發(fā)送的廣播植入IntentIntent intent = new Intent("com.example.activitytest.Activity.LOCAL_BROADCAST");if(Build.VERSION.SDK_INT >= 28){// 突破隱式廣播限制intent.addFlags(0x01000000);}localBroadcastManager.sendBroadcast(intent); // 發(fā)送本地廣播});// 動態(tài)注冊的步驟intentFilter = new IntentFilter();// 添加自定義廣播intentFilter.addAction("com.example.activitytest.Activity.LOCAL_BROADCAST");// 實例化接收器localReceiver = new LocalReceiver();// 注冊接收器localBroadcastManager.registerReceiver(localReceiver, intentFilter);}// 動態(tài)注冊一定要在結(jié)束時取消注冊@Overrideprotected void onDestroy() {super.onDestroy();localBroadcastManager.unregisterReceiver(localReceiver);}class LocalReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context, "本地廣播", Toast.LENGTH_LONG).show();}} }總結(jié)
以上是生活随笔為你收集整理的Android入门(11)| 全局广播与本地广播的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 广发银行ONE卡麦兜联名信用卡年费多少?
- 下一篇: Android开发(4) | 系统权限、