Android zygote啟動流程詳解
在Android系統中,zygote是一個native進程,是所有應用進程的父進程。而zygote則是Linux系統用戶空間的第一個進程——init進程,通過fork的方式創建并啟動的。
作用zygote進程在啟動時,會創建一個Dalvik虛擬機實例,每次孵化新的應用進程時,都會將這個Dalvik虛擬機實例復制到新的應用程序進程里面,從而使得每個應用程序進程都有一個獨立的Dalvik虛擬機實例。
zygote進程的主要作用有兩個:
啟動SystemServer。孵化應用進程。
啟動流程啟動入口Zygote進程在init進程中,通過解析init.zygote.rc配置文件,以service(服務)的方式啟動并創建的。
以init.zygote32.rc為例來看下:
腳本講解// systemcorerootdirinit.zygote32.rcservice zygote /system/bin/app_process -Xzygote /system/bin --zygote --start-system-server class main priority -20 user root group root readproc reserved_disk socket zygote stream 660 root system socket usap_pool_primary stream 660 root system onrestart write /sys/power/state on onrestart restart audioserver onrestart restart cameraserver onrestart restart media onrestart restart netd onrestart restart wificond writepid /dev/cpuset/foreground/tasks
這段腳本要求 init 進程創建一個名為 zygote 的進程,該進程要執行的程序是“/system/bin/app_process”。并且為 zygote 進程創建一個 socket 資源 (用于進程間通信,ActivityManagerService 就是通過該 socket 請求 zygote 進程 fork 一個應用程序進程)。
后面的**--zygote**是參數,表示啟動的是zygote進程。在app_process的main函數中會依據該參數決定執行ZygoteInit還是Java類。
啟動過程zygote要執行的程序便是system/bin/app_process,它的源代碼在frameworks/base/cmds/app_process/app_main.cpp
App_main::mainint main(int argc, char* const argv[]){ ... while (i < argc) { const char* arg = argv[i++]; if (strcmp(arg, '--zygote') == 0) {//是否有--zygote參數。這個是啟動zygote進程的時候的參數 zygote = true;//進程名稱,設置為zygote niceName = ZYGOTE_NICE_NAME; } else if (strcmp(arg, '--start-system-server') == 0) {//是否有--start-system-server startSystemServer = true;.... if (zygote) {//最最重要方法。。。如果是zygote進程,則啟動ZygoteInit。 runtime.start('com.android.internal.os.ZygoteInit', args, zygote); } else if (className) { runtime.start('com.android.internal.os.RuntimeInit', args, zygote); } else { fprintf(stderr, 'Error: no class name or --zygote supplied.n'); app_usage(); LOG_ALWAYS_FATAL('app_process: no class name or --zygote supplied.'); }}AndroidRuntime::start
void AndroidRuntime::start(const char* className, const Vector<String8>& options, bool zygote){ ... JNIEnv* env;//重點方法 創建VM虛擬機,參數是指針,可以用于獲取返回的值,可以使用env來和Java層來做交互 if (startVm(&mJavaVM, &env, zygote) != 0) { return; } onVmCreated(env); //重點方法 給虛擬機注冊一些JNI函數,(系統so庫、用戶自定義so庫 、加載函數等。) if (startReg(env) < 0) { ALOGE('Unable to register all android nativesn'); return; } //找到類的main方法,并調用。如果是zygote的話,這里就會啟動ZygoteInit類的main方法 jmethodID startMeth = env->GetStaticMethodID(startClass, 'main', '([Ljava/lang/String;)V'); if (startMeth == NULL) { ALOGE('JavaVM unable to find main() in ’%s’n', className); /* keep going */ } else { //調用main方法。這里通過JNI調用Java方法之后,Zygote(Native層)就進入了Java的世界,從而開啟了Android中Java的世界。 env->CallStaticVoidMethod(startClass, startMeth, strArray);}
App_main.main AndroidRuntime.start startVm//創建虛擬機 startReg//注冊JNI函數 ZygoteInit.main//這里就進入到了Java層了 registerZygoteSocket//建立IPC的通訊機制 preload//預加載類和資源 startSystemServer//啟動system_server runSelectLoop//等待進程創建的請求
對應的源碼地址: /frameworks/base/cmds/app_process/App_main.cpp (內含AppRuntime類) /frameworks/base/core/jni/AndroidRuntime.cpp /frameworks/base/core/java/com/android/internal/os/ZygoteInit.java /frameworks/base/core/java/com/android/internal/os/Zygote.java /frameworks/base/core/java/android/net/LocalServerSocket.java
Zygote進程的啟動過程中,除了會創建一個Dalvik虛擬機實例之外,還會將Java運行時庫加載到進程中,以及注冊一些Android核心類的JNI方法到創建的Dalvik虛擬機實例中。
zygote進程初始化時啟動虛擬,并加載一些系統資源。這樣zygote fork出子進程之后,子進程也會繼承能正常工作的虛擬機和各種系統資源,剩下的只需要裝載APK文件的字節碼就可以運行程序,。
Java應用程序不能以本地進程的形態運行,必須在一個獨立的虛擬機中運行。如果每次都重新啟動虛擬機,肯定就會拖慢應用程序的啟動速度。
注意:APK應用程序進程被zygote進程孵化出來以后,不僅會獲得Dalvik虛擬機實例拷貝,還會與Zygote一起共享Java運行時庫。
以上就是Android zygote啟動流程詳解的詳細內容,更多關于Android zygote啟動流程的資料請關注好吧啦網其它相關文章!
相關文章: