我们知道Android系统是基于Linux内核的,在Linux系统中所有的进程都是init进程的子进程。Zygote也一样它是在系统启动的过程中由init进程创建的,在系统启动脚本init.rc中:
@init.rc
service zygote /syste/bin/app_process -Xzygote /system/bin -zygote --start-system-server
class main
socket zygote stream 666
onrestart restart surfaceflinger
...
关键字server告诉init进程创建一个名为zygote的进程,执行的的程序是/syste/bin/app_process,后面都是传递给这个进程的参数。下面的socket表示这个zygote进行需要一个名称为zygote的socket套接字资源,系统启动后在/dev/socket目录下可以看到相关的文件。
zygote 服务进程也叫孵化进程,在Linux的用户空间进程app_process会做一些zygote进程启动的前期工作,如启动runtime运行时的环境等,接着调用runtime.start()来执行zygote服务。runtime.start()这个函数实际是类函数AndroidRuntime::start(),会创建并启动一个虚拟机实例来执行com.android.internal.os.ZygoteInit这个包的main函数。这个main函数会fork一个子进程来启动systemServer,父进程就作为真正的孵化器存在了,每当系统要求执行一个Android应用程序,Zygote就会接收到socket消息fork出一个子进程来执行该应用程序。 而每一个Android应用程序都运行在一个Dalvik虚拟机实例里,每一个虚拟机实例都是一个独立的进程空间。
@frameworks/base/cmds/app_process/app_main.cpp
int main(int argc, char* argv[])
{
AppRuntime runtime;// 这里AppRuntime是AndroidRuntime的子类,启动runtime运行时环境。
...// 获取参数
/*
argv[0] = "-Xzygote"
argv[1] = "/system/bin"
argv[2] = "--zygote"
argv[3] = "--start-system-server"
startSystemServer = true
*/
int i = runtime.addVMArguments(argc. argv);// 找到参数第一个不是以单个-开始的参数,这里是/system/bin
runtime.mParentDir = argv[i++]// 将目录保存到mParentDir变量中
setArgsv0(argv0, "zygote")
set_process_name("zygote");
runtime.start("com.android.internal.os.zygoteInit", startSystemServer ? "start-system-server" : "");
}
这里主要就是调用runtime.start() 函数,我们重点看下runtime类型为AppRuntime,它继承于AndroidRuntime类。
@frameworks/base/cmds/app_process/app_main.cpp
class AppRuntime : public AndroidRuntime
{
public:
AppRuntime() : mParentDir(NULL), mClassName(NULL), mClass(NULL), mArgc(0), mArgv(NULL)
{}
...
const char* getClassName() const {}
virtual void onVmCreated(JNIEnv* env) {}
virtual void onStarted() {}
virtual void onZygoteInit() {}
virtual void onExit(int code)
const char* mParentDir;
const char* mClassName;
jclass mClas;
int mArgC;
const char* const* mArgV;
};
@frameworks/base/include/android_runtime/AndroidRuntime.h
@frameworks/base/core/jni/AndroidRuntime.cpp
/* static */ JavaVM* AndroidRuntime::mJavaVM = NULL;
void AndroidRuntime::start(const char* className, const char* options)
{
...
JNIEnv *env;
startVm(&mJavaVM, &env);// start the virtual machine
onVmCreated(env);
startReg(env);// Register android functions
// call className main
char *slashClassName = toSlashClassName(className);
jclass startClass = env->FindClass(slashClassName);
jmethodID startMeth = env->GetStaticMethodID(startClass, "main", "([Ljava/lang/String;]V)");