Android zygote啟動(dòng)流程詳解
在Android系統(tǒng)中,zygote是一個(gè)native進(jìn)程,是所有應(yīng)用進(jìn)程的父進(jìn)程。而zygote則是Linux系統(tǒng)用戶空間的第一個(gè)進(jìn)程——init進(jìn)程,通過(guò)fork的方式創(chuàng)建并啟動(dòng)的。
作用zygote進(jìn)程在啟動(dòng)時(shí),會(huì)創(chuàng)建一個(gè)Dalvik虛擬機(jī)實(shí)例,每次孵化新的應(yīng)用進(jìn)程時(shí),都會(huì)將這個(gè)Dalvik虛擬機(jī)實(shí)例復(fù)制到新的應(yīng)用程序進(jìn)程里面,從而使得每個(gè)應(yīng)用程序進(jìn)程都有一個(gè)獨(dú)立的Dalvik虛擬機(jī)實(shí)例。
zygote進(jìn)程的主要作用有兩個(gè):
啟動(dòng)SystemServer。孵化應(yīng)用進(jìn)程。
啟動(dòng)流程啟動(dòng)入口Zygote進(jìn)程在init進(jìn)程中,通過(guò)解析init.zygote.rc配置文件,以service(服務(wù))的方式啟動(dòng)并創(chuàng)建的。
以init.zygote32.rc為例來(lái)看下:
腳本講解// 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 進(jìn)程創(chuàng)建一個(gè)名為 zygote 的進(jìn)程,該進(jìn)程要執(zhí)行的程序是“/system/bin/app_process”。并且為 zygote 進(jìn)程創(chuàng)建一個(gè) socket 資源 (用于進(jìn)程間通信,ActivityManagerService 就是通過(guò)該 socket 請(qǐng)求 zygote 進(jìn)程 fork 一個(gè)應(yīng)用程序進(jìn)程)。
后面的**--zygote**是參數(shù),表示啟動(dòng)的是zygote進(jìn)程。在app_process的main函數(shù)中會(huì)依據(jù)該參數(shù)決定執(zhí)行ZygoteInit還是Java類。
啟動(dòng)過(guò)程zygote要執(zhí)行的程序便是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參數(shù)。這個(gè)是啟動(dòng)zygote進(jìn)程的時(shí)候的參數(shù) zygote = true;//進(jìn)程名稱,設(shè)置為zygote niceName = ZYGOTE_NICE_NAME; } else if (strcmp(arg, '--start-system-server') == 0) {//是否有--start-system-server startSystemServer = true;.... if (zygote) {//最最重要方法。。。如果是zygote進(jìn)程,則啟動(dòng)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;//重點(diǎn)方法 創(chuàng)建VM虛擬機(jī),參數(shù)是指針,可以用于獲取返回的值,可以使用env來(lái)和Java層來(lái)做交互 if (startVm(&mJavaVM, &env, zygote) != 0) { return; } onVmCreated(env); //重點(diǎn)方法 給虛擬機(jī)注冊(cè)一些JNI函數(shù),(系統(tǒng)so庫(kù)、用戶自定義so庫(kù) 、加載函數(shù)等。) if (startReg(env) < 0) { ALOGE('Unable to register all android nativesn'); return; } //找到類的main方法,并調(diào)用。如果是zygote的話,這里就會(huì)啟動(dòng)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 { //調(diào)用main方法。這里通過(guò)JNI調(diào)用Java方法之后,Zygote(Native層)就進(jìn)入了Java的世界,從而開(kāi)啟了Android中Java的世界。 env->CallStaticVoidMethod(startClass, startMeth, strArray);}
App_main.main AndroidRuntime.start startVm//創(chuàng)建虛擬機(jī) startReg//注冊(cè)JNI函數(shù) ZygoteInit.main//這里就進(jìn)入到了Java層了 registerZygoteSocket//建立IPC的通訊機(jī)制 preload//預(yù)加載類和資源 startSystemServer//啟動(dòng)system_server runSelectLoop//等待進(jìn)程創(chuàng)建的請(qǐng)求
對(duì)應(yīng)的源碼地址: /frameworks/base/cmds/app_process/App_main.cpp (內(nèi)含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進(jìn)程的啟動(dòng)過(guò)程中,除了會(huì)創(chuàng)建一個(gè)Dalvik虛擬機(jī)實(shí)例之外,還會(huì)將Java運(yùn)行時(shí)庫(kù)加載到進(jìn)程中,以及注冊(cè)一些Android核心類的JNI方法到創(chuàng)建的Dalvik虛擬機(jī)實(shí)例中。
zygote進(jìn)程初始化時(shí)啟動(dòng)虛擬,并加載一些系統(tǒng)資源。這樣zygote fork出子進(jìn)程之后,子進(jìn)程也會(huì)繼承能正常工作的虛擬機(jī)和各種系統(tǒng)資源,剩下的只需要裝載APK文件的字節(jié)碼就可以運(yùn)行程序,。
Java應(yīng)用程序不能以本地進(jìn)程的形態(tài)運(yùn)行,必須在一個(gè)獨(dú)立的虛擬機(jī)中運(yùn)行。如果每次都重新啟動(dòng)虛擬機(jī),肯定就會(huì)拖慢應(yīng)用程序的啟動(dòng)速度。
注意:APK應(yīng)用程序進(jìn)程被zygote進(jìn)程孵化出來(lái)以后,不僅會(huì)獲得Dalvik虛擬機(jī)實(shí)例拷貝,還會(huì)與Zygote一起共享Java運(yùn)行時(shí)庫(kù)。
以上就是Android zygote啟動(dòng)流程詳解的詳細(xì)內(nèi)容,更多關(guān)于Android zygote啟動(dòng)流程的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. XML入門精解之結(jié)構(gòu)與語(yǔ)法2. jsp文件下載功能實(shí)現(xiàn)代碼3. React優(yōu)雅的封裝SvgIcon組件示例4. CSS Hack大全-教你如何區(qū)分出IE6-IE10、FireFox、Chrome、Opera5. ASP中格式化時(shí)間短日期補(bǔ)0變兩位長(zhǎng)日期的方法6. jsp+servlet實(shí)現(xiàn)猜數(shù)字游戲7. ASP基礎(chǔ)知識(shí)Command對(duì)象講解8. ASP腳本組件實(shí)現(xiàn)服務(wù)器重啟9. 基于javaweb+jsp實(shí)現(xiàn)企業(yè)財(cái)務(wù)記賬管理系統(tǒng)10. jsp+mysql實(shí)現(xiàn)網(wǎng)頁(yè)的分頁(yè)查詢
