詳解Android廣播機制
謹以文章記錄學習歷程,如有錯誤還請指明。
前言
我們上學時都有過這樣的經歷,當我們在火車站列車候車室中等待時,每當有某次列車開始檢票或者進站上車時,就會播放通知來告知在候車室等待的人們該訊息。
為了便於進行系統級別的訊息通知,Android引入了一套類似的廣播機制,然而比上述情景要靈活得多。此文將對Android廣播機制的方方面面做出詳盡的介紹。
Android廣播機制簡介
前面我們提到,Android的廣播機制更加的靈活,這是因為Android允許每個應用只對自己感興趣的廣播進行註冊,這樣該程式就只會收到自己所關心的廣播內容。
Android廣播分為兩個方面:廣播發送者
BroadcastReceiver
指的就是廣播接收者(廣播接收器)。
應用場景
同一應用具有多個程序的不同元件之間的訊息通訊
- 不同應用間的元件之間的訊息通訊
- 與Android系統在特定情況下的通訊
- 如:系統開機,網路變化等
以上只說明適合廣播機制的應用場景,還有一些場景理論上可以使用,但是實際開發沒有人這麼做:
同一應用內同一組件的訊息通訊:顯然擴充套件變數的作用域、介面回撥、
Handler-Message
等方式都能更簡單的實現。同一應用內的不同元件之間的訊息通訊(單個程序):對於簡單的的情況,依靠介面的回撥方式就可解決;而較為複雜的情況,更推薦直接使用
EventBus
實現原理
設計模式與模型
Android中的廣播使用了觀察者模式,模型為 基於訊息的釋出/訂閱事件模型。
從設計模式上講,廣播的傳送者和接收者極大程度的解耦,使得系統方便整合,容易擴充套件
模型成員:
- 訊息釋出者(廣播發布者)
- 訊息訂閱者(廣播接收者)
- 訊息中心(AMS,Activity Manager Service,一個Android系統中極其重要!的成分,以後我們會詳細講解)
此處我們擴充套件一下,觀察者模式和釋出訂閱模式的關係
釋出訂閱模式屬於廣義上的觀察者模式
前者時最常用的一種觀察者模式的實現,且從解耦和重用角度上看更優於典型的觀察者模式釋出訂閱模式加入訊息中心,實現釋出者和訂閱者的解耦:
- 在觀察者模式中,觀察者需要直接訂閱目標事件,在目標發出內容改變的事件後,直接接收事件並作出響應。
- 在釋出訂閱模式中,多了一個訊息中心,一方面從釋出者接收事件,另一方面向訂閱者釋出事件,訂閱者需要從訊息中心訂閱事件。以此避免釋出者和訂閱者之間產生依賴關係。
實現流程
- 廣播接收者
BroadcastReceiver
通過Binder
機制向AMS(Activity Manager Service
)進行註冊; - 廣播發送者通過
binder
機制向AMS傳送廣播; - AMS查詢符合相應條件(
IntentFilter
/Permission
等)的BroadcastReceiver
- AMS將廣播發送到上述符合條件的
BroadcastReceiver
相應的訊息迴圈佇列中 BroadcastReceiver
通過訊息迴圈執行拿到此廣播,回撥BroadcastReceiver
中的onReceive()
方法。
廣播發送者和廣播接收者的執行是非同步的,發出去的廣播不會關心有無接收者接收,也不確定接收者到底是何時才能接收到。
BroadcastReceiver
自定義BroadcastReceiver
- 繼承基類
BroadcaseReceiver
實現抽象方法
onReceive(context, intent)
- 收到廣播後,會自動回撥
onReceive(..)
方法 - 通常,
onReceive(..)
方法會涉及到與其他元件的互動,如傳送Notification
,啟動service
等 - 預設情況,
BroadcaseReceiver
執行在UI執行緒,因此,onReceive(..)
方法不能執行耗時操作,否則ANR
- 收到廣播後,會自動回撥
簡單的自定義Demo:
MyBroadcastReceiver.java
//繼承BroadcastReceiver基類
public class MyBroadcastReceiver extends BroadcastReceiver {
private static final String TAG = "MyBroadcastReceiver";
@Override
public void onReceive(Context context, Intent intent) {
StringBuilder sb = new StringBuilder();
sb.append("Action: " + intent.getAction() + "\n");
sb.append("URI: " + intent.toUri(Intent.URI_INTENT_SCHEME).toString() + "\n");
String log = sb.toString();
Log.d(TAG, log);
Toast.makeText(context, log, Toast.LENGTH_LONG).show();
}
}
BroadcastReceiver註冊型別
1. 靜態註冊
- 在
AndroidManifest.xml
檔案中通過<receiver>
進行註冊 - 規則及例項說明:
<receiver
//BroadcastReceiver子類的類名
android:name="string"
//是否使用該BroadcastReceiver
android:enabled=["true" | "false"]
//此broadcastReceiver能否接收其他App的發出的廣播
//其預設值是由receiver中有無intent-filter決定的,如果有intent-filter,預設值為true,否則為false
android:exported=["true" | "false"]
android:icon="drawable resource"
android:label="string resource"
//具有相應許可權的廣播發送方傳送的廣播才能被此broadcastReceiver所接收
android:permission="string"
//broadcastReceiver執行所處的程序。
//預設為app的程序,可以指定獨立的程序
//Android四大基本元件都可以通過此屬性指定自己的獨立程序
android:process="string">
//指定此廣播接收器將用於接收特定的廣播型別
//本例中給出的時系統開機後自身發出的廣播
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED"/>
</intent-filter>
</receiver>
以上述靜態方法註冊的MyBroadcastReceiver
,在app
首次啟動時,系統或自動例項化MyBroadcastReceiver
,並註冊到系統中。
2. 動態註冊
- 在程式碼中呼叫
Context.registerReceiver()
, - 典型寫法示例如下:
public class MainActivity extends AppCompatActivity {
public static final String BROADCAST_ACTION = "com.example.whd_alive";
private BroadcastReceiver mBroadcastReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//例項化MyBroadcastReceiver
mBroadcastReceiver = new MyBroadcastReceiver();
//例項化IntentFilter
IntentFilter intentFilter = new IntentFilter();
//設定接收廣播的型別
intentFilter.addAction(BROADCAST_ACTION);
//動態註冊
registerReceiver(mBroadcastReceiver, intentFilter);
}
//銷燬廣播
//當此Activity例項化時,會動態將MyBroadcastReceiver註冊到系統中
//當此Activity銷燬時,動態註冊的MyBroadcastReceiver將不再接收到相應的廣播
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(mBroadcastReceiver);
}
}
注:Android中所有與觀察者模式有關的設計中,一旦涉及到register,必定在相應的時機需要unregister。因此,上例在
onDestroy()
回撥需要unregisterReceiver(mBroadcastReceiver)
。
廣播發送及廣播型別
廣播發送
- 廣播 這一實體本身以
intent
表示 - 廣播的定義 = 相應廣播
intent
的定義 - 廣播的傳送:通過廣播發送者將此
intent
傳送出去,根據不同型別的廣播呼叫相對應的send方法
廣播的型別
主要分為一下四類:
Normal Broadcast
(普通廣播):通常呼叫sendBroadcast(Intent)(Intent, String)
方法傳送System Broadcast
(系統廣播):發生各種事件時,系統自動傳送Ordered Broadcast
(有序廣播):呼叫sendOrderedBroadcast(Intent, String)
方法傳送Local Broadcast
(本地廣播):呼叫LocalBroadcastManager.sendBroadcast(intent)
方法傳送Sticky Broadcast
(粘性廣播):已棄用(API 21)
1. Normal Broadcast(普通廣播)
開發者自定義的intent
,以Context.sendBroadcast()
,Context.sendBroadcastAsUser()
等方法傳送該intent
。
- 傳送示例如下:
Intent intent = new Intent();
intent.setAction(BROADCAST_ACTION);
//最普通的傳送方式
sendBroadcast(intent);
//附帶許可權的傳送方式,宣告此許可權的BroadcastReceiver才能接收此廣播
sendBroadcast(intent,RECEIVER_PREMISSION);
//以下兩種不常見,是因為只有預裝在系統映像中的程式才能使用,否則無法使用
//指明接收人的傳送方式
sendBroadcastAsUser(intent,USER_HANDLER);
//指明接收人以及對應許可權的傳送方式
sendBroadcastAsUser(intent,USER_HANDLER,RECEIVER_PREMISSION);
- 若被註冊了的
BroadCastReceiver
註冊的intentFilter
的action
與上述匹配,則會接收此廣播,且順序是無序的。如果傳送時有相應的許可權要求,則BroadCastReceiver
只有擁有相應的許可權才能接受。
<receiver
android:name=".MyBroadcastReceiver"
android:permission="RECEIVER_PREMISSION">
<intent-filter>
<action android:name="BROADCAST_ACTION"/>
</intent-filter>
</receiver>
2. System Broadcast(系統廣播)
文末提供詳細系統廣播清單,不包含使用說明(位於SDK下boradcast_action.txt ),請自行查詢Google官方文件
Android系統中內建了多個系統廣播,只要涉及到手機的基本操作,基本上都會發出相應的系統廣播。每個系統廣播都具有特定的intent-filter
,其中主要包括具體的action
,系統廣播發出後,將被相應的BroadcastReceiver
接收。系統廣播在系統內部當特定事件發生時,有系統自動發出。
3. Ordered Broadcast(有序廣播))
傳送出去的廣播被
BroadcastReceiver
按照先後循序接收。有序廣播的有序廣播中的“有序”是針對廣播接收者而言的
傳送方式:
- 定義過程與普通廣播一樣,呼叫
sendOrderedBroadcast()
,同樣也有對應的sendOrderedBroadcastAsUser()
方法,只不過同樣針對於預裝在系統映像的應用。
- 定義過程與普通廣播一樣,呼叫
特點
- 按順序接收
- 允許優先順序高的
BroadcastReceiver
截斷廣播。 - 允許優先順序高的
BroadcastReceiver
修改廣播
接受順序
priority
值不同:由大到小排序priority
值相同:動態註冊優於靜態註冊
4. Local Broadcast(本地廣播)
- 可以理解成一種區域性廣播的形式,廣播的傳送者和接收者都同屬於一個App
相比於全域性廣播,本地廣播優勢體現在:
- 安全性更高;
- 更加高效。
引入原因:
- 其他App可能會針對性的發出與當前App
intent-filter
相匹配的廣播,由此導致當前App不斷接收到廣播並處理; - 其他App可以註冊與當前App一致的
intent-filter
用於接收廣播,獲取廣播具體資訊。
- 其他App可能會針對性的發出與當前App
解決方案
- 全域性廣播限制為區域性廣播(本質仍為一個全域性廣播)
- 使用本地廣播
方案1的具體實現:
- 對於同一App內部發送和接收廣播,將
exported
屬性人為設定成false
,使得非本App內部發出的此廣播不被接收; - 在廣播發送和接收時,都增加上相應的
permission
,用於許可權驗證; - 傳送廣播時,指定特定廣播接收器所在的包名,具體是通過
intent.setPackage(packageName)
指定在,這樣此廣播將只會傳送到此包中的App內與之相匹配的有效廣播接收器中。
- 對於同一App內部發送和接收廣播,將
方案2的具體實現:
使用封裝好的LocalBroadcastManager
類。
使用方式上與通常的全域性廣播幾乎相同,只是註冊/取消註冊廣播接收器和傳送廣播時將主調context
變成了LocalBroadcastManager
的單一例項。
對於
LocalBroadcastManager
方式傳送的應用內廣播,只能通過LocalBroadcastManager
動態註冊的ContextReceiver
才有可能接收到(靜態註冊或其他方式動態註冊的ContextReceiver
是接收不到的)
程式碼示例如下:
//例項化MyBroadcastReceiver
mBroadcastReceiver = new MyBroadcastReceiver();
//例項化IntentFilter
IntentFilter intentFilter = new IntentFilter();
//得到LocalBroadcastManager例項
LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);
//設定接收廣播的型別
intentFilter.addAction(BROADCAST_ACTION);
//動態註冊
localBroadcastManager.registerReceiver(mBroadcastReceiver, intentFilter);
//取消註冊
localBroadcastManager.unregisterReceiver(mBroadcastReceiver);
不同註冊方式的廣播接收器回撥onReceive(context, intent)中的context具體型別
- 靜態註冊(全域性+本地):
回撥onReceive(context, intent)
中的context
具體指的是ReceiverRestrictedContext
- 全域性動態註冊:
回撥onReceive(context, intent)
中的context
具體指的是Activity Context
; - LocalBroadcastManager動態註冊
回撥onReceive(context, intent)
中的context
具體指的是Application Context
。
Android 7.0以後的新特性
以上我們討論了老生常談的內容,下面我們談一談Android 7.0以後的新變化
Android 7.0起,系統不再發送以下系統廣播:
ACTION_NEW_PICTURE
ACTION_NEW_VIDEO
針對Android 7.0 (API級別24)和更高版本的應用程式必須通過
registerReceiver()
註冊以下廣播。在AndroidManifest
中宣告<receiver>
起作用。CONNECTIVITY_ACTION
Android 8.0起,應用無法在
Manifest
中註冊大部分隱式系統廣播(即,並非專門針對此應用的廣播),此意也是在於降低隨Android同時執行的應用增多,發生效能變差的機率。
出於安全考慮的廣播使用最佳實踐
如不需要嚮應用程式之外的元件傳送廣播,則可以使用支援庫
Support Library
中LocalBroadcastManager
傳送和接收本地廣播。如果許多應用程式清單中註冊接收相同的廣播,它會導致系統啟動大量的應用程式,從而對裝置效能和使用者體驗產生重大影響。為了避免這種情況,請使用動態註冊而不是
Manifest
宣告。有時,Android系統本身會強制使用上下文註冊的接收器。例如,CONNECTIVITY_ACTION
廣播只允許動態註冊。onReceive(Context, Intent)
執行在UI執行緒,不要進行耗時操作- 如耗時操作必不可少,生成子執行緒。
不要使用隱含的意圖傳播敏感資訊。這些資訊可以被任何註冊的應用程式讀取。
- 解決方案 :
permission
/setPackage(String)
/LocalBroadcastManager
.
- 解決方案 :
當註冊一個
BroadcastReceiver
,任何應用程式都可以傳送潛在的惡意廣播到你的應用的BroadcastReceiver
。- 解決方案 :
permission
/android:exported = "false"
/LocalBroadcastManager
.
- 解決方案 :
廣播操作的名稱空間是全域性的。確保操作名稱和其他字串都是在您自己的名稱空間中編寫的,否則您可能會無意中與其他應用程式發生衝突。
不要從
BroadcastReceiver
開始活動,這麼做會導致使用者體驗很差,特別是如果有不止一個BroadcastReceiver
。相反,考慮使用Notification
。
附錄
Android 系統廣播清單
android.accounts.LOGIN_ACCOUNTS_CHANGED
android.accounts.action.ACCOUNT_REMOVED
android.app.action.ACTION_PASSWORD_CHANGED
android.app.action.ACTION_PASSWORD_EXPIRING
android.app.action.ACTION_PASSWORD_FAILED
android.app.action.ACTION_PASSWORD_SUCCEEDED
android.app.action.APPLICATION_DELEGATION_SCOPES_CHANGED
android.app.action.APP_BLOCK_STATE_CHANGED
android.app.action.DEVICE_ADMIN_DISABLED
android.app.action.DEVICE_ADMIN_DISABLE_REQUESTED
android.app.action.DEVICE_ADMIN_ENABLED
android.app.action.DEVICE_OWNER_CHANGED
android.app.action.INTERRUPTION_FILTER_CHANGED
android.app.action.LOCK_TASK_ENTERING
android.app.action.LOCK_TASK_EXITING
android.app.action.NEXT_ALARM_CLOCK_CHANGED
android.app.action.NOTIFICATION_CHANNEL_BLOCK_STATE_CHANGED
android.app.action.NOTIFICATION_CHANNEL_GROUP_BLOCK_STATE_CHANGED
android.app.action.NOTIFICATION_POLICY_ACCESS_GRANTED_CHANGED
android.app.action.NOTIFICATION_POLICY_CHANGED
android.app.action.PROFILE_OWNER_CHANGED
android.app.action.PROFILE_PROVISIONING_COMPLETE
android.app.action.SYSTEM_UPDATE_POLICY_CHANGED
android.appwidget.action.APPWIDGET_DELETED
android.appwidget.action.APPWIDGET_DISABLED
android.appwidget.action.APPWIDGET_ENABLED
android.appwidget.action.APPWIDGET_HOST_RESTORED
android.appwidget.action.APPWIDGET_RESTORED
android.appwidget.action.APPWIDGET_UPDATE
android.appwidget.action.APPWIDGET_UPDATE_OPTIONS
android.bluetooth.a2dp.profile.action.CONNECTION_STATE_CHANGED
android.bluetooth.a2dp.profile.action.PLAYING_STATE_CHANGED
android.bluetooth.adapter.action.CONNECTION_STATE_CHANGED
android.bluetooth.adapter.action.DISCOVERY_FINISHED
android.bluetooth.adapter.action.DISCOVERY_STARTED
android.bluetooth.adapter.action.LOCAL_NAME_CHANGED
android.bluetooth.adapter.action.SCAN_MODE_CHANGED
android.bluetooth.adapter.action.STATE_CHANGED
android.bluetooth.device.action.ACL_CONNECTED
android.bluetooth.device.action.ACL_DISCONNECTED
android.bluetooth.device.action.ACL_DISCONNECT_REQUESTED
android.bluetooth.device.action.BOND_STATE_CHANGED
android.bluetooth.device.action.CLASS_CHANGED
android.bluetooth.device.action.FOUND
android.bluetooth.device.action.NAME_CHANGED
android.bluetooth.device.action.PAIRING_REQUEST
android.bluetooth.device.action.UUID
android.bluetooth.devicepicker.action.DEVICE_SELECTED
android.bluetooth.devicepicker.action.LAUNCH
android.bluetooth.headset.action.VENDOR_SPECIFIC_HEADSET_EVENT
android.bluetooth.headset.profile.action.AUDIO_STATE_CHANGED
android.bluetooth.headset.profile.action.CONNECTION_STATE_CHANGED
android.bluetooth.hiddevice.profile.action.CONNECTION_STATE_CHANGED
android.bluetooth.input.profile.action.CONNECTION_STATE_CHANGED
android.bluetooth.pan.profile.action.CONNECTION_STATE_CHANGED
android.bluetooth.pbap.profile.action.CONNECTION_STATE_CHANGED
android.content.pm.action.SESSION_COMMITTED
android.hardware.action.NEW_PICTURE
android.hardware.action.NEW_VIDEO
android.hardware.hdmi.action.OSD_MESSAGE
android.hardware.input.action.QUERY_KEYBOARD_LAYOUTS
android.hardware.usb.action.USB_ACCESSORY_ATTACHED
android.hardware.usb.action.USB_ACCESSORY_DETACHED
android.hardware.usb.action.USB_DEVICE_ATTACHED
android.hardware.usb.action.USB_DEVICE_DETACHED
android.intent.action.ACTION_POWER_CONNECTED
android.intent.action.ACTION_POWER_DISCONNECTED
android.intent.action.ACTION_SHUTDOWN
android.intent.action.AIRPLANE_MODE
android.intent.action.APPLICATION_RESTRICTIONS_CHANGED
android.intent.action.BATTERY_CHANGED
android.intent.action.BATTERY_LOW
android.intent.action.BATTERY_OKAY
android.intent.action.BOOT_COMPLETED
android.intent.action.CAMERA_BUTTON
android.intent.action.CLOSE_SYSTEM_DIALOGS
android.intent.action.CONFIGURATION_CHANGED
android.intent.action.CONTENT_CHANGED
android.intent.action.DATA_SMS_RECEIVED
android.intent.action.DATE_CHANGED
android.intent.action.DEVICE_STORAGE_LOW
android.intent.action.DEVICE_STORAGE_OK
android.intent.action.DOCK_EVENT
android.intent.action.DOWNLOAD_COMPLETE
android.intent.action.DOWNLOAD_NOTIFICATION_CLICKED
android.intent.action.DREAMING_STARTED
android.intent.action.DREAMING_STOPPED
android.intent.action.DROPBOX_ENTRY_ADDED
android.intent.action.EXTERNAL_APPLICATIONS_AVAILABLE
android.intent.action.EXTERNAL_APPLICATIONS_UNAVAILABLE
android.intent.action.FACTORY_RESET
android.intent.action.FETCH_VOICEMAIL
android.intent.action.GTALK_CONNECTED
android.intent.action.GTALK_DISCONNECTED
android.intent.action.HEADSET_PLUG
android.intent.action.HEADSET_PLUG
android.intent.action.INPUT_METHOD_CHANGED
android.intent.action.INTENT_FILTER_NEEDS_VERIFICATION
android.intent.action.LOCALE_CHANGED
android.intent.action.LOCKED_BOOT_COMPLETED
android.intent.action.MANAGE_PACKAGE_STORAGE
android.intent.action.MASTER_CLEAR_NOTIFICATION
android.intent.action.MEDIA_BAD_REMOVAL
android.intent.action.MEDIA_BUTTON
android.intent.action.MEDIA_CHECKING
android.intent.action.MEDIA_EJECT
android.intent.action.MEDIA_MOUNTED
android.intent.action.MEDIA_NOFS
android.intent.action.MEDIA_REMOVED
android.intent.action.MEDIA_SCANNER_FINISHED
android.intent.action.MEDIA_SCANNER_SCAN_FILE
android.intent.action.MEDIA_SCANNER_STARTED
android.intent.action.MEDIA_SHARED
android.intent.action.MEDIA_UNMOUNTABLE
android.intent.action.MEDIA_UNMOUNTED
android.intent.action.MY_PACKAGE_REPLACED
android.intent.action.NEW_OUTGOING_CALL
android.intent.action.NEW_VOICEMAIL
android.intent.action.PACKAGES_SUSPENDED
android.intent.action.PACKAGES_UNSUSPENDED
android.intent.action.PACKAGE_ADDED
android.intent.action.PACKAGE_CHANGED
android.intent.action.PACKAGE_DATA_CLEARED
android.intent.action.PACKAGE_FIRST_LAUNCH
android.intent.action.PACKAGE_FULLY_REMOVED
android.intent.action.PACKAGE_INSTALL
android.intent.action.PACKAGE_NEEDS_VERIFICATION
android.intent.action.PACKAGE_REMOVED
android.intent.action.PACKAGE_REPLACED
android.intent.action.PACKAGE_RESTARTED
android.intent.action.PACKAGE_VERIFIED
android.intent.action.PHONE_STATE
android.intent.action.PROVIDER_CHANGED
android.intent.action.PROXY_CHANGE
android.intent.action.QUERY_PACKAGE_RESTART
android.intent.action.REBOOT
android.intent.action.SCREEN_OFF
android.intent.action.SCREEN_ON
android.intent.action.SIM_STATE_CHANGED
android.intent.action.TIMEZONE_CHANGED
android.intent.action.TIME_SET
android.intent.action.TIME_TICK
android.intent.action.UID_REMOVED
android.intent.action.UMS_CONNECTED
android.intent.action.UMS_DISCONNECTED
android.intent.action.USER_PRESENT
android.intent.action.USER_UNLOCKED
android.intent.action.WALLPAPER_CHANGED
android.media.ACTION_SCO_AUDIO_STATE_UPDATED
android.media.AUDIO_BECOMING_NOISY
android.media.RINGER_MODE_CHANGED
android.media.SCO_AUDIO_STATE_CHANGED
android.media.VIBRATE_SETTING_CHANGED
android.media.action.CLOSE_AUDIO_EFFECT_CONTROL_SESSION
android.media.action.HDMI_AUDIO_PLUG
android.media.action.MICROPHONE_MUTE_CHANGED
android.media.action.OPEN_AUDIO_EFFECT_CONTROL_SESSION
android.media.tv.action.CHANNEL_BROWSABLE_REQUESTED
android.media.tv.action.INITIALIZE_PROGRAMS
android.media.tv.action.PREVIEW_PROGRAM_ADDED_TO_WATCH_NEXT
android.media.tv.action.PREVIEW_PROGRAM_BROWSABLE_DISABLED
android.media.tv.action.WATCH_NEXT_PROGRAM_BROWSABLE_DISABLED
android.net.conn.BACKGROUND_DATA_SETTING_CHANGED
android.net.conn.CONNECTIVITY_CHANGE
android.net.conn.RESTRICT_BACKGROUND_CHANGED
android.net.nsd.STATE_CHANGED
android.net.scoring.SCORER_CHANGED
android.net.scoring.SCORE_NETWORKS
android.net.wifi.NETWORK_IDS_CHANGED
android.net.wifi.RSSI_CHANGED
android.net.wifi.SCAN_RESULTS
android.net.wifi.STATE_CHANGE
android.net.wifi.WIFI_STATE_CHANGED
android.net.wifi.aware.action.WIFI_AWARE_STATE_CHANGED
android.net.wifi.p2p.CONNECTION_STATE_CHANGE
android.net.wifi.p2p.DISCOVERY_STATE_CHANGE
android.net.wifi.p2p.PEERS_CHANGED
android.net.wifi.p2p.STATE_CHANGED
android.net.wifi.p2p.THIS_DEVICE_CHANGED
android.net.wifi.rtt.action.WIFI_RTT_STATE_CHANGED
android.net.wifi.supplicant.CONNECTION_CHANGE
android.net.wifi.supplicant.STATE_CHANGE
android.nfc.action.ADAPTER_STATE_CHANGED
android.nfc.action.TRANSACTION_DETECTED
android.os.action.DEVICE_IDLE_MODE_CHANGED
android.os.action.POWER_SAVE_MODE_CHANGED
android.provider.Telephony.SECRET_CODE
android.provider.Telephony.SIM_FULL
android.provider.Telephony.SMS_CB_RECEIVED
android.provider.Telephony.SMS_DELIVER
android.provider.Telephony.SMS_RECEIVED
android.provider.Telephony.SMS_REJECTED
android.provider.Telephony.SMS_SERVICE_CATEGORY_PROGRAM_DATA_RECEIVED
android.provider.Telephony.WAP_PUSH_DELIVER
android.provider.Telephony.WAP_PUSH_RECEIVED
android.provider.action.DEFAULT_SMS_PACKAGE_CHANGED
android.provider.action.EXTERNAL_PROVIDER_CHANGE
android.provider.action.SYNC_VOICEMAIL
android.security.STORAGE_CHANGED
android.security.action.KEYCHAIN_CHANGED
android.security.action.KEY_ACCESS_CHANGED
android.security.action.TRUST_STORE_CHANGED
android.speech.tts.TTS_QUEUE_PROCESSING_COMPLETED
android.speech.tts.engine.TTS_DATA_INSTALLED
android.telephony.action.DEFAULT_SMS_SUBSCRIPTION_CHANGED
android.telephony.action.DEFAULT_SUBSCRIPTION_CHANGED
android.telephony.action.REFRESH_SUBSCRIPTION_PLANS
android.telephony.action.SIM_APPLICATION_STATE_CHANGED
android.telephony.action.SIM_CARD_STATE_CHANGED
android.telephony.action.SIM_SLOT_STATUS_CHANGED
android.telephony.action.SUBSCRIPTION_CARRIER_IDENTITY_CHANGED
android.telephony.euicc.action.NOTIFY_CARRIER_SETUP
android.telephony.euicc.action.OTA_STATUS_CHANGED
總結
- 本文全面的介紹Android中的廣播機制,對廣播的傳送,接受,以及Android新版本的變化等方方面面做出詳細的總結。
- 筆者水平有限,如有錯漏,歡迎指正。
- 不定期分享Android開發相關的技術乾貨,期待與你的交流,共勉。