Android應用程式架構層建立的應用程式進程具有兩個特點,一是進程的入口函數是ActivityThread.main,二是進程天然支援 Binder處理序間通訊機制;這兩個特點都是在進程的初始化過程中實現的,本文將詳細分析Android應用程式進程建立過程中是如何?這兩個特點的。
Android應用程式架構層建立的應用程式進程的入口函數是ActivityThread.main比較好理解,即進程建立完成之 後,Android應用程式架構層就會在這個進程中將ActivityThread類載入進來,然後執行它的main函數,這個main函數就是進程執行 訊息迴圈的地方了。Android應用程式架構層建立的應用程式進程天然支援Binder處理序間通訊機制這個特點應該怎麼樣理解呢?前面我們在學習 Android系統的Binder處理序間通訊機制時說到,它具有四個組件,分別是驅動程式、守護進程、Client以及Server,其中Server組 件在初始化時必須進入一個迴圈中不斷地與Binder驅動程式進行到互動,以便獲得Client組件發送的請求,具體可參考Android系統處理序間通訊 (IPC)機制Binder中的Server啟動過程原始碼分析一文,但是,當我們在Android應用程式中實現Server組件的時候,我們並沒有讓 進程進入一個迴圈中去等待Client組件的請求,然而,當Client組件得到這個Server組件的遠程介面時,卻可以順利地和Server組件進行 處理序間通訊,這就是因為Android應用程式進程在建立的時候就已經啟動了一個線程池來支援Server組件和Binder驅動程式之間的互動了,這 樣,極大地方便了在Android應用程式中建立Server組件。
在Android應用程式架構層中,是由ActivityManagerService組件負責為Android應用程式建立新的進程的,它本來也是 運行在一個獨立的進程之中,不過這個進程是在系統啟動的過程中建立的。ActivityManagerService組件一般會在什麼情況下會為應用程式 建立一個新的進程呢?當系統決定要在一個新的進程中啟動一個Activity或者Service時,它就會建立一個新的進程了,然後在這個新的進程中啟動 這個Activity或者Service,具體可以參考Android系統在新進程中啟動自訂服務過程(startService)的原理分析、 Android應用程式啟動過程原始碼分析和Android應用程式在新的進程中啟動新的Activity的方法和過程分析這三篇文章。
ActivityManagerService啟動新的進程是從其成員函數startProcessLocked開始的,在深入分析這個過程之前,我們先來看一下進程建立過程的順序圖表,然後再詳細分析每一個步驟。
Step 1. ActivityManagerService.startProcessLocked
這個函數定義在frameworks/base/services/java/com/android/server/am/ActivityManagerService.java檔案中:
- [java] view plaincopypublic final class ActivityManagerService extends
- ActivityManagerNative
- implements Watchdog.Monitor, BatteryStatsImpl.BatteryCallback {
- ......
- private final void startProcessLocked(ProcessRecord app,
- String hostingType, String hostingNameStr) {
- ......
- try {
- int uid = app.info.uid;
- int[] gids = null;
- try {
- gids = mContext.getPackageManager().getPackageGids(
- app.info.packageName);
- } catch (PackageManager.NameNotFoundException e) {
- ......
- }
- ......
- int debugFlags = 0;
- ......
- int pid = Process.start("android.app.ActivityThread",
- mSimpleProcessManagement ? app.processName : null, uid, uid,
- gids, debugFlags, null);
- ......
- } catch (RuntimeException e) {
- ......
- }
- }
- ......
- }
它調用了Process.start函數開始為應用程式建立新的進程,注意,它傳入一個第一個參數 為"android.app.ActivityThread",這就是進程初始化時要載入的Java類了,把這個類載入到進程之後,就會把它裡面的靜態成 員函數main作為進程的進入點,後面我們會看到。