日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android入门(11)| 全局广播与本地广播

發(fā)布時間:2023/12/13 Android 42 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android入门(11)| 全局广播与本地广播 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

文章目錄

  • 廣播概念
  • 接收廣播
    • 動態(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):

public class BroadcastActivity extends AppCompatActivity {private IntentFilter intentFilter; // 意圖過濾器,旨在匹配可以響應(yīng)對應(yīng)操作的組件private NetworkChangeReceiver networkChangeReceiver; // 自定義的廣播接收器類@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);intentFilter = new IntentFilter();// android.net.conn.CONNECTIVITY_CHANGE是網(wǎng)絡(luò)狀態(tài)變化時系統(tǒng)的廣播intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");networkChangeReceiver = new NetworkChangeReceiver();// 注冊,將接收器與IntentFilter相匹配registerReceiver(networkChangeReceiver, intentFilter);}// 動態(tài)注冊一定要在結(jié)束時取消注冊protected void onDestroy(){super.onDestroy();// 實現(xiàn)取消注冊unregisterReceiver(networkChangeReceiver);}// 自定義的廣播接收器類class NetworkChangeReceiver extends BroadcastReceiver{@Overridepublic void onReceive(Context context, Intent intent) {// 通過getSystemService得到管理網(wǎng)絡(luò)連接的connectivityManager實例ConnectivityManager connectivityManager = (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();// 通過isAvailable方法判斷是否有網(wǎng)絡(luò)if(networkInfo != null && networkInfo.isAvailable()){Toast.makeText(context, "network is available", Toast.LENGTH_SHORT).show();} else{Toast.makeText(context, "network is unavailable", Toast.LENGTH_SHORT).show();}}} }

靜態(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ù)傳遞:

public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 收到自定義廣播時會彈出提示Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();// 截斷abortBroadcast();} }

不必對另一個接收器 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)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。