一、Dalvik虚拟机启动

在启动Zygote进程时,会启动Dalvik虚拟机,完毕以下几件事:

1. 创建了一个Dalvik虚拟机实例;

2. 载入了Java核心类及注冊其JNI方法;

3. 为主线程的设置了一个JNI环境;

4. 注冊了Android核心类的JNI方法。

void AndroidRuntime::start(const char* className, const bool startSystemServer)
{
...... /* start the virtual machine */
if (startVm(&mJavaVM, &env) != 0)
goto bail; /*
* Register android functions.
*/
if (startReg(env) < 0) {
LOGE("Unable to register all android natives\n");
goto bail;
} ...... /*
* Start VM. This thread becomes the main thread of the VM, and will
* not return until the VM exits.
*/
jclass startClass;
jmethodID startMeth; slashClassName = strdup(className);
for (cp = slashClassName; *cp != '\0'; cp++)
if (*cp == '.')
*cp = '/'; startClass = env->FindClass(slashClassName);
if (startClass == NULL) {
LOGE("JavaVM unable to locate class '%s'\n", slashClassName);
/* keep going */
} else {
startMeth = env->GetStaticMethodID(startClass, "main",
"([Ljava/lang/String;)V");
if (startMeth == NULL) {
LOGE("JavaVM unable to find main() in '%s'\n", className);
/* keep going */
} else {
env->CallStaticVoidMethod(startClass, startMeth, strArray);
......
}
} LOGD("Shutting down VM\n");
if (mJavaVM->DetachCurrentThread() != JNI_OK)
LOGW("Warning: unable to detach main thread\n");
if (mJavaVM->DestroyJavaVM() != 0)
LOGW("Warning: VM did not shut down cleanly\n"); ......
}

二、第一次解释运行java程序。env->CallStaticVoidMethod(startClass, startMeth, strArray)终于会解释运行com.android.internal.os.ZygoteInit类的静态成员函数main

參考Dalvik虚拟机的运行过程分析。随着方法调用的不同,不断运行不同的GOTO_TARGET(invokeMethod, bool methodCallRange, const Method* _methodToCall, u2 count, u2 regs)或者GOTO_TARGET(returnFromMethod)等等。去解释运行java程序。

三、Zygote进程等待来在SystemServer的请求,来创建应用程序进程

static native int selectReadable(FileDescriptor[] fds) throws IOException;

此时dalvik虚拟机不是解释运行java程序,而是调用C/C++程序的入口地址,直接运行C/C++代码。因为在第一步中已经注冊Android核心JNI,所以能够直接调用,请看考以下的第六点。

四、Zygote进程创建SystemServer进程

pid = Zygote.forkSystemServer(
parsedArgs.uid, parsedArgs.gid,
parsedArgs.gids, debugFlags, null,
parsedArgs.permittedCapabilities,
parsedArgs.effectiveCapabilities);
 if (pid == 0) {
handleSystemServerProcess(parsedArgs);//运行了com.android.server.SystemServer中main方法
}

本质上使用fork来创建一个新的进程。Zygote进程在启动是创建的Dalvik虚拟机实例和主线程的环境,使用COW的方式共享。其它两点因为是仅仅读,共享就可以。

返回到pid==0,此时已经是SystemServer进程的Davlik虚拟机主线程的环境来解释运行handleSystemServerProcess。

五、SystemServer进程调用Process.start。通过Socket请求Zygote进程来创建应用程序进程

pid = Zygote.forkAndSpecialize(parsedArgs.uid, parsedArgs.gid,
parsedArgs.gids, parsedArgs.debugFlags, rlimits);
if (pid == 0) {
// in child
handleChildProc(parsedArgs, descriptors, newStderr);//android.app.ActivityThread中的main函数
// should never happen
return true;
}

和第四点的描写叙述一样。

六、应用程序进程注冊JNI。并运行JNI方法。

注冊过程请參考Dalvik虚拟机JNI方法的注冊过程分析,会调用JNI_OnLoad获取主线程的Dalvik虚拟机环境JNIENV,并调用了dvmSetNativeFunc设置了navtiveFunc。

当运行到JNI方法时。会调用*method->nativeFunc。不是解释运行,是直接在机器上运行。

void dvmCallMethodV(Thread* self, const Method* method, Object* obj,
bool fromJni, JValue* pResult, va_list args)
{
...... if (dvmIsNativeMethod(method)) {
TRACE_METHOD_ENTER(self, method);
/*
* Because we leave no space for local variables, "curFrame" points
* directly at the method arguments.
*/
(*method->nativeFunc)(self->curFrame, pResult, method, self);
TRACE_METHOD_EXIT(self, method);
} else {
dvmInterpret(self, method, pResult);
} ......
}

当假设想在JNI方法中调用Java方法,必须得有当前线程的Davlik虚拟机环境JNIENV,比如:

(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);

和第二点一样,必须由Davlik虚拟机去解释运行Java方法。

七、应用程序进程,Java中开启一个线程,Thread.start

參考Dalvik虚拟机进程和线程的创建过程分析,使用clone创建了一个新的线程。和进程(主线程)共享进程空间。因为此线程没有Dalvik虚拟机环境JNIENV,所以要为该进程设置虚拟机环境。有了环境后就能够在解释运行Java代码了。例如以下:

dvmCallMethod(self, run, self->threadObj, &unused);  

八、应用程序进程。C/C++中开启一个线程
    1、.仅仅运行C/C++代码的Native线程的创建过程:

不会创建该线程的Dalvik虚拟机环境JNIENV。

2、能同一时候运行C/C++代码和Java代码的Native线程的创建过程

须要创建该线程的Dalvik虚拟机环境JNIENV,就能够解释运行Java代码了。

Dalvik虚拟机总结的更多相关文章

  1. Android ART运行时与Dalvik虚拟机

    这几天在做一个项目时需要在Android中使用OSGi框架(Apache Felix),于是在一个android 4.4.2 版本系统的某品牌的平板上实验. 实验内容很简单:把felix包里的feli ...

  2. java虚拟机和Dalvik虚拟机的区别

    java虚拟机和Dalvik虚拟机的区别: java虚拟机Dalvik虚拟机 java虚拟机基于栈. 基于栈的机器必须使用指令来载入和操作栈上数据,所需指令更多更多dalvik虚拟机是基于寄存器的 j ...

  3. 转 Android Dalvik虚拟机初识

    首先,让我们来思考下面几个问题: 什么是Dalvik虚拟机? Dalvik VM与JVM有什么区别? Dalvik VM有什么新的特点? Dalvik VM的架构是怎么样的? 首先,我得承认第一个问题 ...

  4. Android Dalvik虚拟机初识(转)

    原文地址:http://blog.csdn.net/andyxm/article/details/6126907 android虚拟机jvmjava优化linux内核 首先,让我们来思考下面几个问题: ...

  5. Android(java)学习笔记156:Java虚拟机和Dalvik虚拟机的区别

    Google于2007年底正式发布了Android SDK, 作为 Android系统的重要特性,Dalvik虚拟机也第一次进入了人们的视野.它对内存的高效使用,和在低速CPU上表现出的高性能,确实令 ...

  6. Dalvik虚拟机进程和线程的创建过程分析

    从前面Dalvik虚拟机的运行过程分析一文可以知道,Dalvik虚拟机除了可以执行Java代码之外,还可以执行Native代码,也就是C/C++函数. 这些C/C++函数在执行的过程中,又可以通过本地 ...

  7. Dalvik虚拟机的运行过程分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/8914953 在前面一篇文章中,我们分析了Dal ...

  8. Dalvik虚拟机JNI方法的注册过程分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/8923483 在前面一文中,我们分析了Dalvi ...

  9. Dalvik虚拟机简要介绍和学习计划

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/8852432 我们知道,Android应用程序是 ...

  10. Dalvik虚拟机的启动过程分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/8885792 在Android系统中,应用程序进 ...

随机推荐

  1. 各种轮播实现(纯css实现+js实现)

    1.纯Css实现轮播效果 <!DOCTYPE html> <html lang="en"> <head> <meta charset=&q ...

  2. [译]The multi Interface

    The multi Interfacemulti接口 The easy interface as described in detail in this document is a synchrono ...

  3. 简单谈谈MySQL中的int(m)

    转载地址:https://www.jb51.net/article/93760.htm 设置int型的时候,需要设置int(M),以前知道这个M最大是255,但是到底应该设置多少并没有在意.注意zer ...

  4. redis+mysql读写方案

    前言:在web服务端开发的过程中,redis+mysql是最常用的存储解决方案,mysql存储着所有的业务数据,根据业务规模会采用相应的分库分表.读写分离.主备容灾.数据库集群等手段.但是由于mysq ...

  5. http请求和响应头部

    说一说常见的请求头和响应头都有什么呢? 1)请求(客户端->服务端[request])     GET(请求的方式) /newcoder/hello.html(请求的目标资源) HTTP/1.1 ...

  6. Rxjava1升级Rxjava2踩坑一记

    Rxjava1升级Rxjava2坑 共存问题 通常情况下,如果我们希望在一个模块中既想使用rxjava1又想使用rxjava2,这个时候在运行的时候会出现一下报错: ... APK META/-INF ...

  7. STM32的串口DMA收发以及双缓冲区的实现

    在使用STM32的UART的DMA功能总结如下: 首先上代码,这里采用STM32 的USART1作为Demo,RX的DMA为DMA1_Channel5,TX的DMA为DMA1_Channel4.初始化 ...

  8. JavaScriptav数据类型和变量

    数据类型 计算机顾名思义就是可以做数学计算的机器,因此,计算机程序理所当然地可以处理各种数值.但是,计算机能处理的远不止数值,还可以处理文本.图形.音频.视频.网页等各种各样的数据,不同的数据,需要定 ...

  9. vue组件---边界处理情况

    (1)访问元素&组件 ①访问根实例 在每个 new Vue 实例的子组件中,其根实例可以通过 $root 属性进行访问.例如,在这个根实例中: // Vue 根实例 new Vue({ dat ...

  10. vue启动

    首先在终端terminal连上npm 镜像库 npm config set registry https://registry.npm.taobao.orgnpm installnpm run loc ...