Android zygote启动流程详解

 更新时间:2021年03月30日 08:56:33   作者:kailasa  
这篇文章主要介绍了Android zygote启动流程的相关资料,帮助大家更好的理解和学习使用Android开发,感兴趣的朋友可以了解下

对zygote的理解

在Android系统中,zygote是一个native进程,是所有应用进程的父进程。而zygote则是Linux系统用户空间的第一个进程——init进程,通过fork的方式创建并启动的。

作用

zygote进程在启动时,会创建一个Dalvik虚拟机实例,每次孵化新的应用进程时,都会将这个Dalvik虚拟机实例复制到新的应用程序进程里面,从而使得每个应用程序进程都有一个独立的Dalvik虚拟机实例。

zygote进程的主要作用有两个:

启动SystemServer。
孵化应用进程。

启动流程

启动入口

Zygote进程在init进程中,通过解析init.zygote.rc配置文件,以service(服务)的方式启动并创建的。

以init.zygote32.rc为例来看下:

脚本讲解

//  system\core\rootdir\init.zygote32.rc
service 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::main

int 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 natives\n");
    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启动流程的资料请关注脚本之家其它相关文章!

相关文章

  • android Handler详细使用方法实例

    android Handler详细使用方法实例

    本文主要介绍Android中Handler的简单使用方法,Handler跟多线程,消息队列联系很紧密,在平常的实际程序开发中比较常见。本文分为4个简单的例子来学校handler
    2013-11-11
  • Android 7.0应用之间如何共享文件

    Android 7.0应用之间如何共享文件

    这篇文章主要介绍了Android 7.0应用之间如何共享文件,帮助大家更好的理解和使用Android进行开发,感兴趣的朋友可以了解下
    2020-12-12
  • Android使用fastjson库解析json字符串实战

    Android使用fastjson库解析json字符串实战

    fastjson是一个Java语言编写的高性能功能完善的JSON库,它采用一种“假定有序快速匹配”的算法,把JSON Parse的性能提升到极致,是目前Java语言中最快的JSON库,Fastjson接口简单易用,已经被广泛使用在缓存序列化、协议交互、Web输出、Android客户端等多种应用场景
    2023-11-11
  • Android编程实现Toast只显示最后一条的方法

    Android编程实现Toast只显示最后一条的方法

    这篇文章主要介绍了Android编程实现Toast只显示最后一条的方法,结合实例形式总结了Toast只显示最后一条的原理与具体实现技巧,需要的朋友可以参考下
    2017-08-08
  • 使用Android Studio 开发自己的SDK教程

    使用Android Studio 开发自己的SDK教程

    很多时候我们要将自己开发一个类库打包成jar包以供他调用,这个jar包也叫你自己的SDK或者叫library。android studio生成jar包的方法与eclipse有所不同。在studio中library其实是module的概念。
    2017-10-10
  • android socket聊天室功能实现

    android socket聊天室功能实现

    这篇文章主要为大家详细介绍了android socket聊天室功能实现方法,不单纯是聊天室,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • Android自定义wheelview随机选号效果

    Android自定义wheelview随机选号效果

    这篇文章主要介绍了Android自定义wheelview随机选号效果,利用wheelview实现滚动随机选择号码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • 详解Android性能优化之启动优化

    详解Android性能优化之启动优化

    一款应用的第一印象很重要,第一印象往往决定了用户的去留。打开一款应用,如果速度很快,很顺畅,那么很容易让人觉得这款应用背后的技术实力很强,用户潜意识中会对这款应用更加的信赖。本文将详细介绍Android性能优化之启动优化。
    2021-06-06
  • Android编程中HTTP服务用法详解

    Android编程中HTTP服务用法详解

    这篇文章主要介绍了Android编程中HTTP服务用法,结合实例形式较为详细的分析了Android中http服务的具体步骤与相关使用技巧,需要的朋友可以参考下
    2016-01-01
  • Android中EditText setText方法的踩坑实战

    Android中EditText setText方法的踩坑实战

    这篇文章主要给大家分享了一些关于Android中EditText setText方法的踩坑记录,文中通过示例代码介绍的非常详细,对各位Android开发者们具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-07-07

最新评论