1.Zygote是什么?
2.Zygote的启动流程?
3.Zygote的工作原理?
虽然做android很多年了,但真正看源码的时候还是在4.0的时候,现在想要系统的回顾一下,才发现对android系统还是一头雾水,所以打算系统学习一下Framework。
Zygote的作用呢,也很简单,就是启动SystemServer和孵化应用进程。Zygote会初始化一些常用类,JNI函数,预加载主题资源和共享库。SystemServer需要用到Zygote初始化的这些东西,所以直接从Zygote继承过来自己就不用重新加载一遍了。
Linux启动之后用户空间的第一个进程是Init进程,它会加载init.rc这个启动配置文件,Zygote就是init.rc中要启动的进程之一,还有像ServiceManager进程等。
进程启动有2种方式,一种是fork+handle(子进程继承父进程资源),另一种是fork+execve,2者都会返回2次,即子进程返回一次并且返回的pid==0,fork+execve这种方式在子进程中会执行execve函数,传入的参数有3个分别为可执行程序的路径,参数,环境变量。
fork+execve-> pid_t pid = fork();
if(pid == 0){ // child process
execve(path,argv,env);
}else{ // parent process }
信号处理之SIGCHLD,父进程fork出子进程,如果子进程挂了的话,它的父进程就会收到一个SIGCHLD的信号,父进程就会知道子进程挂了,就会重启子进程。
Zygote的native世界:
启动Android虚拟机 -> 注册Android的JNI函数->进入Java世界 (Native切换到Java是如何实现的)
都知道我们的应用程序是跑在jvm上的,但这个jvm我们是何时创建的呢?
其实我们的应用程序是Zygote孵化的,Zygote在初始化的时候就已经创建好了jvm,应用程序就继承了它的jvm,只需要在应用启动时重置一下jvm的状态和重启一下jvm的守护线程即可。
Zygote的Java世界:
1.Preload Resources 预加载资源,为将来孵化子进程时可以共享给它们的。
2.fork-> SystemServer (SystemServer是单独跑在一个进程里的)
3.进入Loop循环,接收并处理socket通信,runOnce函数(读取参数列表,孵化子进程,执行子进程main函数)
Zygote fork时是单线程的,在fork子进程时会停掉其它线程,fork结束后会重启其它线程
Zygote的IPC没有采用binder机制而是本地socket。
Binder基于 Client-Server通信模式,传输过程只需一次拷贝,为发送发添加UID/PID身份,既支持实名Binder也支持匿名Binder,安全性高。Binder通讯是需要多线程操作的,代理对象对Binder的调用是在Binder线程,需要再通过Handler调用主线程来操作。
根据UNIX上C++程序设计守则3,多线程程序里不准使用fork,原因嘛就是会带来死锁,具体怎么造成的请查设计守则,里面有详细说明。