1. 客戶端query關鍵點
按以前的分析習慣,碰到Binder調用時會馬上轉入服務端(即Bn端)去分析,但是這個思路在query函數中行不通。為什么?來看IContentProvider Bp端的query函數,它定義在ContentProviderProxy中,代碼如下:
**ContentProviderNative.java::ContentProviderProxy.query**
~~~
public Cursor query(Uri url, String[] projection,String selection,
String[] selectionArgs, String sortOrder) throws RemoteException {
//①構造一個BulkCursorToCursorAdaptor對象
BulkCursorToCursorAdaptor adaptor = new BulkCursorToCursorAdaptor();
Parceldata = Parcel.obtain();
Parcel reply = Parcel.obtain();
try {
data.writeInterfaceToken(IContentProvider.descriptor);
......//將參數打包到data請求包中
//②adaptor.getObserver()返回一個IContentObserver類型的對象,把它
//也打包到data請求包中。ContentObserver相關的知識留到第8章再分析
data.writeStrongBinder(adaptor.getObserver().asBinder());
//發送請求給遠端的Bn端
mRemote.transact(IContentProvider.QUERY_TRANSACTION, data, reply, 0);
DatabaseUtils.readExceptionFromParcel(reply);
//③從回復包中得到一個IBulkCursor類型的對象
IBulkCursor bulkCursor =
BulkCursorNative.asInterface(reply.readStrongBinder());
if(bulkCursor != null) {
int rowCount = reply.readInt();
int idColumnPosition = reply.readInt();
boolean wantsAllOnMoveCalls = reply.readInt() != 0;
//④調用adaptor的initialize函數
adaptor.initialize(bulkCursor, rowCount,
idColumnPosition, wantsAllOnMoveCalls);
} ......
return adaptor;
} ......
finally {
data.recycle();
reply.recycle();
}
}
~~~
從以上代碼中可發現,ContentProviderProxy query函數中還大有文章,其中一共列出了4個關鍵點。最令人頭疼的是其中新出現的兩個類BulkCursorToCursorAdaptor和IBulkCursor。此處不必急于分析它們。
我們再到服務端去提取query函數的關鍵點。
2. 服務端query關鍵點
根據第2章對Binder的介紹,客戶端發來的請求先在Bn端的onTransact中得到處理,代碼如下:
**ContentProviderNative.java::onTransact**
~~~
public boolean onTransact(int code, Parcel data, Parcelreply, int flags)
throws RemoteException {
try {
switch (code) {
case QUERY_TRANSACTION://處理query請求
{
data.enforceInterface(IContentProvider.descriptor);
Uri url = Uri.CREATOR.createFromParcel(data);
......//從請求包中提取參數
//⑤創建ContentObserver Binder通信的Bp端
IContentObserver observer = IContentObserver.Stub.asInterface(
data.readStrongBinder());
//⑥調用MediaProvider實現的query函數。Cursor是一個接口類,那么變量
//cursor的真實類型是什么?
Cursor cursor = query(url, projection, selection, selectionArgs,
sortOrder);
if(cursor != null) {
//⑦創建一個CursorToBulkCursorAdaptor類型的對象
CursorToBulkCursorAdaptor adaptor = new
CursorToBulkCursorAdaptor(
cursor,observer, getProviderName());
final IBinder binder = adaptor.asBinder();
//⑧返回結果集所含記錄項的條數,這個函數看起來極不起眼,但卻非常為關鍵
final int count = adaptor.count();
//返回名為"_id"的列在結果集中的索引位置,該列由數據庫在創建表時自動添加
final int index = BulkCursorToCursorAdaptor.findRowIdColumnIndex(
adaptor.getColumnNames());
//wantsAllOnMoveCalls一般為false,讀者閱讀完本章后再自行分析它
final boolean wantsAllOnMoveCalls =
adaptor.getWantsAllOnMoveCalls();
reply.writeNoException();
//將binder的信息寫到reply回復包中
reply.writeStrongBinder(binder);
reply.writeInt(count);//將結果集包含的記錄項行數返回給客戶端
reply.writeInt(index);
reply.writeInt(wantsAllOnMoveCalls ? 1 : 0);
}......
return true;
}......//其他情況處理
......
}
~~~
和客戶端對應,服務端的query處理也比較復雜,其中的攔路虎仍是新出現的幾種數據類型。
在掃清這些攔路虎之前,應先把客戶端和服務端query調用的關鍵點總結一下。
3. 提取query關鍵點總結
我們提取query兩端的調用關鍵點,如圖7-5所示。
:-: 
圖7-5 query調用關鍵點示意
再來總結一下前面提到的幾個攔路虎,它們分別是:
- 客戶端創建的BulkCursorToCursorAdaptor、從服務端query后返回的IBulkCursor。
- 服務端創建的CursorToCursorAdaptor,以及從子類query函數返回的Cursor。
從名字上看,這幾個類都和Cursor有關,所以有必要先搞清MediaProviderquery返回的Cursor到底是什么。
* * * * *
**注意**:圖7-5借用了UML的序列圖來展示query調用順序,其中ContentProvider框和MediaProviderer框代表同一個對象。另外,圖7-5中的調用函數編號并不完全對應代碼中的關鍵函數調用編號。
* * * * *
- 前言
- 第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 本章小結