前面在介紹PKMS掃描APK時提到,PKMS將解析得到的Package私有的Activity信息加入到自己的數據結構mActivities中保存。先來回顧一下代碼:
**PacakgeManagerService.java::scanPackageLI函數**
~~~
......//此時APK文件已經解析完成
N =pkg.activities.size();//取出該APK中包含的Activities信息
r =null;
for (i=0; i<N; i++) {
PackageParser.Activity a = pkg.activities.get(i);
a.info.processName = fixProcessName(pkg.applicationInfo.processName,
a.info.processName,pkg.applicationInfo.uid);
mActivities.addActivity(a,"activity");//①加到mActivities中保存
}
~~~
上面的代碼中有兩個比較重要的數據結構,如圖4-11所示。
:-: 
圖4-11 相關數據結構示意圖
結合代碼,由圖4-11可知:
- mActivities為ActivityIntentResolver類型,是PKMS的成員變量,用于保存系統中所有與Activity相關的信息。此數據結構內部有一個mActivities變量,它以ComponetName為Key,保存PackageParser.Activity對象
- 從APK中解析得到的所有和Activity相關的信息(包括在XML中聲明的IntentFilter標簽)都由PacakgeParser.Activity來保存。
前面代碼中調用addActivity函數完成了私有信息的公有化。addActivity函數的代碼如下:
**PacakgeManagerService.java::ActivityIntentResolver.addActivity**
~~~
public final voidaddActivity(PackageParser.Activity a, String type) {
finalboolean systemApp = isSystemApp(a.info.applicationInfo);
//將Component和Activity保存到mActivities中
mActivities.put(a.getComponentName(), a);
finalint NI = a.intents.size();
for(int j=0; j<NI; j++) {
//ActivityIntentInfo存儲的就是XML中聲明的IntentFilter信息
PackageParser.ActivityIntentInfo intent = a.intents.get(j);
if(!systemApp && intent.getPriority() > 0 &&"activity".equals(type)) {
//非系統APK的priority必須為0。后續分析中將介紹priority的作用
intent.setPriority(0);
}
addFilter(intent);//接下來將分析這個函數
}
}
~~~
下面來分析addFilter函數,這里涉及較多的復雜數據結構,代碼如下:
**IntentResolver.java::IntentResolver.addFilter**
~~~
public void addFilter(F f) {
......
mFilters.add(f);//mFilters保存所有IntentFilter信息
//除此之外,為了加快匹配工作的速度,還需要分類保存IntentFilter信息
//下邊register_xxx函數的最后一個參數用于打印信息
intnumS = register_intent_filter(f, f.schemesIterator(),
mSchemeToFilter," Scheme: ");
intnumT = register_mime_types(f, " Type: ");
if(numS == 0 && numT == 0) {
register_intent_filter(f, f.actionsIterator(),
mActionToFilter," Action: ");
}
if(numT != 0) {
register_intent_filter(f, f.actionsIterator(),
mTypedActionToFilter, " TypedAction: ");
}
}
~~~
正如代碼注釋中所說,為了加快匹配工作的速度,這里使用了泛型編程并定義了較多的成員變量。下面總結一下這些變量的作用(注意,除mFilters為HashSet<F>類型外,其他成員變量的類型都是`HashMap<String, ArrayList<F>>`,其中F為模板參數)。
- mSchemeToFilter:用于保存URI中與schema相關的IntentFilter信息。
- mActionToFilter:用于保存僅設置Action條件的IntentFilter信息。
- TypedActionToFilter:用于保存既設置了Action又設置了Data的MIME類型的IntentFilter信息。
- mFilters:用于保存所有IntentFilter信息
- mWildTypeToFilter:用于保存設置了Data類型類似“image/*”的IntentFilter,但是設置MIME類型類似“Image/jpeg”的不算在此類。
- mTypeToFilter:除了包含mWildTypeToFilter外,還包含那些指明了Data類型為確定參數的IntentFilter信息,例如“image/*”和”image/jpeg“等都包含在mTypeToFilter中。
- mBaseTypeToFilter:包含MIME中Base 類型的IntentFilter信息,但不包括Sub type為“*”的IntentFilter。
不妨舉個例子來說明這些變量的用法。
假設,在XML中聲明一個IntentFilter,代碼如下:
~~~
<intent-filter android:label="test">
<actionandroid:name="android.intent.action.VIEW" />
dataandroid:mimeType="audio/*" android:scheme="http"
</intent-filter>
~~~
那么:
- 在mTypedActionToFilter中能夠以“android.intent.action.VIEW”為key找到該IntentFilter。
- 在mWildTypeToFilter和mTypeToFilter中能夠以“audio”為key找到該IntentFilter。
- 在mSchemeToFilter中能夠以”http“為key找到該IntentFilter。
下面來分析Intent匹配查詢工作。
- 前言
- 第1章 搭建Android源碼工作環境
- 1.1 Android系統架構
- 1.2 搭建開發環境
- 1.2.1 下載源碼
- 1.2.2 編譯源碼
- 1.2.3 利用Eclipse調試system_process
- 1.3 本章小結
- 第2章 深入理解Java Binder和MessageQueue
- 2.1 概述
- 2.2 Java層中的Binder架構分析
- 2.2.1 Binder架構總覽
- 2.2.2 初始化Java層Binder框架
- 2.2.3 addService實例分析
- 2.2.4 Java層Binder架構總結
- 2.3 心系兩界的MessageQueue
- 2.3.1 MessageQueue的創建
- 2.3.2 提取消息
- 2.3.3 nativePollOnce函數分析
- 2.3.4 MessageQueue總結
- 2.4 本章小結
- 第3章 深入理解SystemServer
- 3.1 概述
- 3.2 SystemServer分析
- 3.2.1 main函數分析
- 3.2.2 Service群英會
- 3.3 EntropyService分析
- 3.4 DropBoxManagerService分析
- 3.4.1 DBMS構造函數分析
- 3.4.2 dropbox日志文件的添加
- 3.4.3 DBMS和settings數據庫
- 3.5 DiskStatsService和DeviceStorageMonitorService分析
- 3.5.1 DiskStatsService分析
- 3.5.2 DeviceStorageManagerService分析
- 3.6 SamplingProfilerService分析
- 3.6.1 SamplingProfilerService構造函數分析
- 3.6.2 SamplingProfilerIntegration分析
- 3.7 ClipboardService分析
- 3.7.1 復制數據到剪貼板
- 3.7.2 從剪切板粘貼數據
- 3.7.3 CBS中的權限管理
- 3.8 本章小結
- 第4章 深入理解PackageManagerService
- 4.1 概述
- 4.2 初識PackageManagerService
- 4.3 PKMS的main函數分析
- 4.3.1 構造函數分析之前期準備工作
- 4.3.2 構造函數分析之掃描Package
- 4.3.3 構造函數分析之掃尾工作
- 4.3.4 PKMS構造函數總結
- 4.4 APK Installation分析
- 4.4.1 adb install分析
- 4.4.2 pm分析
- 4.4.3 installPackageWithVerification函數分析
- 4.4.4 APK 安裝流程總結
- 4.4.5 Verification介紹
- 4.5 queryIntentActivities分析
- 4.5.1 Intent及IntentFilter介紹
- 4.5.2 Activity信息的管理
- 4.5.3 Intent 匹配查詢分析
- 4.5.4 queryIntentActivities總結
- 4.6 installd及UserManager介紹
- 4.6.1 installd介紹
- 4.6.2 UserManager介紹
- 4.7 本章學習指導
- 4.8 本章小結
- 第5章 深入理解PowerManagerService
- 5.1 概述
- 5.2 初識PowerManagerService
- 5.2.1 PMS構造函數分析
- 5.2.2 init分析
- 5.2.3 systemReady分析
- 5.2.4 BootComplete處理
- 5.2.5 初識PowerManagerService總結
- 5.3 PMS WakeLock分析
- 5.3.1 WakeLock客戶端分析
- 5.3.2 PMS acquireWakeLock分析
- 5.3.3 Power類及LightService類介紹
- 5.3.4 WakeLock總結
- 5.4 userActivity及Power按鍵處理分析
- 5.4.1 userActivity分析
- 5.4.2 Power按鍵處理分析
- 5.5 BatteryService及BatteryStatsService分析
- 5.5.1 BatteryService分析
- 5.5.2 BatteryStatsService分析
- 5.5.3 BatteryService及BatteryStatsService總結
- 5.6 本章學習指導
- 5.7 本章小結
- 第6章 深入理解ActivityManagerService
- 6.1 概述
- 6.2 初識ActivityManagerService
- 6.2.1 ActivityManagerService的main函數分析
- 6.2.2 AMS的 setSystemProcess分析
- 6.2.3 AMS的 installSystemProviders函數分析
- 6.2.4 AMS的 systemReady分析
- 6.2.5 初識ActivityManagerService總結
- 6.3 startActivity分析
- 6.3.1 從am說起
- 6.3.2 AMS的startActivityAndWait函數分析
- 6.3.3 startActivityLocked分析
- 6.4 Broadcast和BroadcastReceiver分析
- 6.4.1 registerReceiver流程分析
- 6.4.2 sendBroadcast流程分析
- 6.4.3 BROADCAST_INTENT_MSG消息處理函數
- 6.4.4 應用進程處理廣播分析
- 6.4.5 廣播處理總結
- 6.5 startService之按圖索驥
- 6.5.1 Service知識介紹
- 6.5.2 startService流程圖
- 6.6 AMS中的進程管理
- 6.6.1 Linux進程管理介紹
- 6.6.2 關于Android中的進程管理的介紹
- 6.6.3 AMS進程管理函數分析
- 6.6.4 AMS進程管理總結
- 6.7 App的 Crash處理
- 6.7.1 應用進程的Crash處理
- 6.7.2 AMS的handleApplicationCrash分析
- 6.7.3 AppDeathRecipient binderDied分析
- 6.7.4 App的Crash處理總結
- 6.8 本章學習指導
- 6.9 本章小結
- 第7章 深入理解ContentProvider
- 7.1 概述
- 7.2 MediaProvider的啟動及創建
- 7.2.1 Context的getContentResolver函數分析
- 7.2.2 MediaStore.Image.Media的query函數分析
- 7.2.3 MediaProvider的啟動及創建總結
- 7.3 SQLite創建數據庫分析
- 7.3.1 SQLite及SQLiteDatabase家族
- 7.3.2 MediaProvider創建數據庫分析
- 7.3.3 SQLiteDatabase創建數據庫的分析總結
- 7.4 Cursor 的query函數的實現分析
- 7.4.1 提取query關鍵點
- 7.4.2 MediaProvider 的query分析
- 7.4.3 query關鍵點分析
- 7.4.4 Cursor query實現分析總結
- 7.5 Cursor close函數實現分析
- 7.5.1 客戶端close的分析
- 7.5.2 服務端close的分析
- 7.5.3 finalize函數分析
- 7.5.4 Cursor close函數總結
- 7.6 ContentResolver openAssetFileDescriptor函數分析
- 7.6.1 openAssetFileDescriptor之客戶端調用分析
- 7.6.2 ContentProvider的 openTypedAssetFile函數分析
- 7.6.3 跨進程傳遞文件描述符的探討
- 7.6.4 openAssetFileDescriptor函數分析總結
- 7.7 本章學習指導
- 7.8 本章小結
- 第8章 深入理解ContentService和AccountManagerService
- 8.1 概述
- 8.2 數據更新通知機制分析
- 8.2.1 初識ContentService
- 8.2.2 ContentResovler 的registerContentObserver分析
- 8.2.3 ContentResolver的 notifyChange分析
- 8.2.4 數據更新通知機制總結和深入探討
- 8.3 AccountManagerService分析
- 8.3.1 初識AccountManagerService
- 8.3.2 AccountManager addAccount分析
- 8.3.3 AccountManagerService的分析總結
- 8.4 數據同步管理SyncManager分析
- 8.4.1 初識SyncManager
- 8.4.2 ContentResolver 的requestSync分析
- 8.4.3 數據同步管理SyncManager分析總結
- 8.5 本章學習指導
- 8.6 本章小結