謹以文章記錄學習歷程,如有錯誤還請指明。
前言
- 我們上學時都有過這樣的經歷,當我們在火車站列車候車室中等待時,每當有某次列車開始檢票或者進站上車時,就會播放通知來告知在候車室等待的人們該消息。
- 為了便于進行系統級別的消息通知,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線程,不要進行耗時操作- 如耗時操作必不可少,生成子線程。
經評論提醒,這里補充一下
具體做法是在 onReceive() 中開啟一個 service,將耗時操作置于 service 中(子線程) -
不要使用隱含的意圖傳播敏感信息。這些信息可以被任何注冊的應用程序讀取。
- 解決方案 :
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新版本的變化等方方面面做出詳細的總結。
- 筆者水平有限,如有錯漏,歡迎指正。
- 接下來我也會將所學的知識分享出來,有興趣可以繼續關注whd_Alive的Android開發筆記
歡迎關注whd_Alive的簡書
- 不定期分享Android開發相關的技術干貨,期待與你的交流,共勉。