发布网友 发布时间:2022-10-18 18:35
共1个回答
热心网友 时间:2023-11-24 04:50
Zygote可以说是Android开发面试很高频的一道问题, 但总有小伙伴在回答这道问题总不能让面试满意, 在这你就要搞清楚面试问你对Zygote的理解时,他最想听到的和其实想问的应该是哪些? 下面我们通过以下几点来剖析这道问题!
Zygote的作用分为两点:
关于这个问题答出了这两点那就是OK了。可能大部分小伙伴可能能答出第二点,第一点就不是很清楚。SystemServer也是Zygote启动的,因为SystemServer需要用到Zygote准备好的系统资源包括:
直接从Zygote继承过来就不需要重新加载过来,那么对性能将会有很大的提升。
在说Zygote启动流程之前, 先明确一个概念:启动三段式, 这个可以理解为Android中进程启动的常用套路, 分为三步骤:
这里要了解LOOP循环是什么,其实LOOP作用是不停地 接受消息 , 处理消息 ,消息的来源可以是 Soket 、 MessageQueue 、 Binder 驱动发过来的消息,但无论消息从哪里来,它整个流程都是去接受消息,处理消息。这个启动三段式,它不光是Zygote进程是这样的,只要是有独立进程的,比如说系统服务进程,自己的应用进程都是如此。
Zygote进程的启动取决于init进程,init进程是它是linux启动之后用户空间的第一个进程,下面看一下 启动流程 :
3.启动配置文件中定义的系统服务,其中Zygote服务就是定义在配置中的
4.同时启动的服务除了Zygote之外还有一些别的系统服务也是会启动的,比如说ServiceManager进程,它是通过fork+execve系统调用启动的
在init.rc 文件中会import /init.${ro.zygote}.rc,init.zygoteXX,XX指的是32或者64,对我们没差我们直接看init.zygote32.rc即可。配置文件比较长,这里做了截取保留了Zygot相关的部分。
说完了启动配置呢,这里来聊一下启动进程, 启动进程有两种方式:
第一种:fork+handle
第二种:fork+execve
两者看起来差不多,首先首先都会调用fork函数创建子进程,这个函数比较奇特会返回两次,子进程返回一次,父进程返回一次。 区别在于:
主要分为两部分Native层处理和Java层处理,Zygote进程启动之后,它执行了execve系统调用,它执行的是用C++写的二进制的可执行程序里的main函数作为入口,然后在Java层运行!
先来看一下Native层的处理流程
在app_main.cpp文件,AndroidRuntime.cpp文件。我们可以找到几个主要函数名
根据上述代码,你会发现在我们的应用里直接就可以 JNI 调用了,并不需要创建虚拟机。因为应用进程是Zygote进程孵化出来的,继承了父进程的拥有虚拟机,只需要重置数据即可。
接着看一下Java层的处理,具体可参考ZygoteInit文件的main方法
1.预加载资源,比如常用类库、主题资源及一些共享库等
2.启动SystemServer进程
3.进入Socket 的Loop循环 会看到的ZygoteServer.runSelectLoop(…)调用
Zygote启动流程中需要主要以下2点问题
最后