JavaVM和JNIEnv的初始化和JVM各模块的初始化都是在JNI_CreateJavaVM()函数中完成。这一篇将详细介绍JavaVM和JNIEnv的初始化过程。

1、初始化JavaVM

JavaVM的初始化都是在JNI_CreateJavaVM()函数中完成,调用链如下:

JavaMain()            java.c
InitializeJVM() java.c
JNI_CreateJavaVM() jni.cpp

在JavaMain()函数中调用InitializeJVM()函数,InitializeJVM()函数会初始化JVM,给JavaVM和JNIEnv变量赋值,通过InvocationFunctions结构体下的CreateJavaVM函数指针来调用对应的函数。在执行LoadJavaVM()函数时,CreateJavaVM函数指针被指向为libjvm.so(在Linux系统上的动态链接库为libjvm.so)动态链接库中JNI_CreateJavaVM()函数。JNI_CreateJavaVM()函数的实现如下:

源代码位置:openjdk/hotspot/src/share/vm/jni.cpp

_JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_CreateJavaVM(JavaVM **vm, void **penv, void *args) {

  // ...
jint result = JNI_ERR; bool can_try_again = true;
//完成JVM的初始化,如果初始化过程中出现不可恢复的异常则can_try_again会被置为false
result = Threads::create_vm((JavaVMInitArgs*) args, &can_try_again); //初始化正常
if (result == JNI_OK) {
//获取当前线程,即执行create_vm()函数的线程,也是JNI_CreateJavaVM()函数执行完毕后执行main()方法的线程
JavaThread *thread = JavaThread::current(); // JavaVM赋值,main_vm是jni.h中的全局变量,最张会指向全局变量jni_InvokeInterface
*vm = (JavaVM *)(&main_vm); // vm是方法的参数,是个双重指针 //JNIEnv赋值,从这里也可以看出,JNIEnv其实是线程私有有
*(JNIEnv**)penv = thread->jni_environment(); // ...
}
// ...
return result;
}

调用Threads::create_vm()函数会初始化一系列的JVM模块,如果函数顺利执行完,则函数会返回JNI_OK,表示正确创建出了JVM实例,我们可以给vm赋值,这样JNI的本地函数就可以通过vm来管理这个创建出来的JVM实例了。

使用main_vm来初始化JavaVM类型的变量,main_vm变量的定义如下:

源代码来源:openjdk/hotspot/src/share/vm/jni.cpp

struct JavaVM_ main_vm = {  &jni_InvokeInterface };

JavaVM是JavaVM_的别名,所以我们后面提到的JavaVM指的就是JavaVM​_。同理,JNIEnv是JNIE​nv_的别名。​

jni_InvokeInterface的定义如下:

const struct JNIInvokeInterface_   jni_InvokeInterface = {
NULL,
NULL,
NULL, jni_DestroyJavaVM,
jni_AttachCurrentThread,
jni_DetachCurrentThread,
jni_GetEnv,
jni_AttachCurrentThreadAsDaemon
};

结构体中保存了一系列函数指针,我们大前面说的JNI的本地函数通过vm来管理JVM实例,其实就是通过函数指针来调用对应的函数管理JVM实例。

通过 JavaVM,我们还可以获得 JVMTI 的指针,并获得 JVMTI 函数的使用能力,所有的 JVMTI 函数都通过jvmtiEnv获取,不同的虚拟机实现提供的函数细节可能不一样,但是使用的方式是统一的,关于JVMTI及相关知识这里不介绍,有兴趣的可自行研究。 

2、初始化JNIEnv

JavaMain()                 java.c
InitializeJVM() java.c
JNI_CreateJavaVM() jni.cpp
Threads::create_vm() thread.cpp
JavaThread::JavaThread() thread.cpp
JavaThread::initialize() thread.cpp
jni_functions() jni.cpp

其中调用的Threads:create_vm()函数中会创建JavaThread实例,如下:

JavaThread* main_thread = new JavaThread()

根据调用链可知,在JavaThread构造函数中会调用JavaThread::initialize()函数,最终会调用jni_functions()函数。jni_functions()函数会返回thread.cpp文件中定义的一个全局变量,如下:

struct JNINativeInterface_* jni_functions() {
return &jni_NativeInterface;
}

jni_NativeInterface这个全局变量的定义如下:

struct JNINativeInterface_  jni_NativeInterface = {
NULL,f
NULL,
NULL,
NULL, jni_GetVersion,
jni_DefineClass,
jni_FindClass, ...
};

如上的结构体中保存了各个JNI函数的指针。

调用jni_functions()函数后还会调用如下函数:

void set_jni_functions(struct JNINativeInterface_* functionTable) {
_jni_environment.functions = functionTable;
}

其中_jni_environment变量的定义如下:

源代码位置:openjdk/hotspot/src/share/vm/rumtime/thread.hpp

// 定义在JavaThread中的变量,所以JNIEnv是线程私有的
JNIEnv _jni_environment;

JNIEnv这个结构体定义在之前说过,在这个结构体中首先定义的就是一个指向JNINativeInterface_的指针functions,如下:

struct JNIEnv_ {
const struct JNINativeInterface_ *functions;
...
}

_jni_environment就是main_thread保留的JNIEnv_实例了,正确设置functions属性就表示该实例初始化完成。

为了让大家一目了然,我简单画了一个示意图。

我们将创建出来的JNIEnv实例保存在线程的_jni_environment变量中,这样当线程执行本地函数时,会从这个变量中取出JNIEnv并传递给本地函数,我们得到JNIEnv后就能调用其中的函数来获取JVM内部提供的服务了,例如想要根据类名来查找jclass,可以调用FindClass()函数。

公众号 深入剖析Java虚拟机HotSpot 已经更新虚拟机源代码剖析相关文章到60+,欢迎关注,如果有任何问题,可加作者微信mazhimazh,拉你入虚拟机群交流

  

  

  

  

  

  

第41篇-JNIEnv与JavaVM的初始化的更多相关文章

  1. 第40篇-JNIEnv和JavaVM

    下面介绍2个与JNI机制相关的类型JNIEnv和JavaVM. 1.JNIEnv JNIEnv一般是是由虚拟机传入,而且与线程相关的变量,也就说线程A不能使用线程B的JNIEnv.而作为一个结构体,它 ...

  2. NDK(13)JNIEnv和JavaVM

    转自:  http://www.cnblogs.com/canphp/archive/2012/11/13/2768937.html JNIEnv是一个与线程相关的变量,不同线程的JNIEnv彼此独立 ...

  3. 【Android JNI】JNIEnv和JavaVM的区别

     JNI的实现可涉及两个关键类:JNIEnv和JavaVM. JavaVM:这个代表java的虚拟机.所有的工作都是从获取虚拟机的接口开始的.             第一种方式,在加载动态链接库的时 ...

  4. Jni 线程JNIEnv,JavaVM,JNI_OnLoad(GetEnv返回NULL?FindClass返回NULL?)

    此文章是关于NDK线程的第二篇理论知识笔记.主要有两个点,如下: 1.pthread_create(Too many arguements, expected 1) ?2.线程中如何获取JNIEnv? ...

  5. Java中JNI的使用详解第二篇:JNIEnv类型和jobject类型的解释

    上一篇说的是一个简单的应用,说明JNI是怎么工作的,这一篇主要来说一下,那个本地方法sayHello的参数的说明,以及其中方法的使用 首先来看一下C++中的sayHello方法的实现: JNIEXPO ...

  6. Java中JNI的使用详解第三篇:JNIEnv类型中方法的使用

    转自: http://blog.csdn.net/jiangwei0910410003/article/details/17466369 上一篇说道JNIEnv中的方法的用法,这一篇我们就来通过例子来 ...

  7. spring boot 学习番外篇:超快速项目初始化

    超快速完成 Spring Boot 项目初始化 最近,在浏览 SPRING 官网时,发现一个超级方便的小工具,可以帮助我们快速创建一个 Spring Boot 项目,前提就是你能连接互联网. 依赖 支 ...

  8. 第41篇 推荐一个jekyll博客模板

    本人用的模板是基于Codeboy的博客模板改造模板,(由于本人可能会有很多样式修改,所以不再将修改pullrequst到原项目,在此对codeboy模板表示感谢).功能改造如下: 添加微信支付宝打赏 ...

  9. NDK(5) Android JNI官方综合教程[JavaVM and JNIEnv,Threads ,jclass, jmethodID, and jfieldID,UTF-8 and UTF-16 Strings,Exceptions,Native Libraries等等]

    JNI Tips In this document JavaVM and JNIEnv Threads jclass, jmethodID, and jfieldID Local and Global ...

随机推荐

  1. MySQL复习(二)MySQL基本数据类型

    MySQL基本数据类型 常用的字段类型大致可以分为数值类型.字符串类型.日期时间类型三大类 1. 数值类型 数值类型可以分为整型.浮点型.定点型三小类. 1.1 整型 (tiny:极小的, small ...

  2. [软工顶级理解组] Beta阶段团队贡献分评分

    贡献分评分依据 下述表格适用于前端.后端.爬虫开发者的评分,在此基础上进行增减. 类别 程度 加减分 准时性 提前完成 +0 按时完成 +0 延后完成,迟交时间一天内或未延误进度 -2 延后完成,迟交 ...

  3. [对对子队]会议记录5.14(Scrum Meeting1)

    今天已完成的工作 何瑞 ​ 工作内容:初步完成循环指令系统 ​ 相关issue:实现循环语句系统的逻辑 ​ 相关签入:feat:循环语句的指令编辑系统初步完成 吴昭邦 ​ 工作内容:将流水线系统和循环 ...

  4. [no code][scrum meeting] Alpha 15

    项目 内容 会议时间 2020-04-23 会议主题 OCR紧急会议 会议时长 45min 参会人员 PM + OCR组(赵涛,黎正宇) 项目 内容 会议时间 2020-04-24 会议主题 全体测试 ...

  5. 使用flink实现一个简单的wordcount

    使用flink实现一个简单的wordcount 一.背景 二.需求 三.前置条件 1.jdk版本要求 2.maven版本要求 四.实现步骤 1.创建 flink 项目 2.编写程序步骤 1.创建Str ...

  6. 大牛针对零基础入门c语言详解指针(超详细)

    C语言指针说难不难但是说容易又是最容易出错的地方,因此不管是你要做什么只要用到C指针你就跳不过,今天咱们就以 十九个例子来给大家简单的分析一下指针的应用,最后会有C语言视频资料提供给大家更加深入的参考 ...

  7. [WPF] 在 Windows 11 中处理 WindowChrome 的圆角

    1. Windows 11 的圆角 在直角统治了微软的 UI 设计多年以后,微软突然把直角骂了一顿,说还是圆角好看,于是 Windows 11 随处都可看到圆角设计.Windows 11 使用 3 个 ...

  8. 从0到1使用Kubernetes系列(五):Kubernetes Scheduling

    前述文章介绍了Kubernetes基本介绍,搭建Kubernetes集群所需要的工具,如何安装,如何搭建应用.本篇介绍怎么使用Kubernetes进行资源调度. Kubernetes作为一个容器编排调 ...

  9. cf12D Ball(MAP,排序,贪心思想)

    题意: N位女士一起聚在一个舞厅.每位女士有三个特征值B,I,R.分别代表美貌,智慧,富有. 对于一位女士而言,如果存在一个女士的B,I,R都分别大于她自己的B,I,R.则她自己会自杀. 统计总共有多 ...

  10. Docker安装配置Tomcat

    1.使用docker pull tomcat下载镜像(不加tag则是下载最新版本) 2.运行容器(-d 后台运行:-p 指定端口映射),接的是镜像ID 3.进入容器执行命令,接的是容器ID 4.宿主机 ...