举一个native方法调用解释执行的Java方法的实例,如下:

  1. public class TestJNI {
  2. static {
  3. System.load("/media/mazhi/sourcecode/workspace/projectjava/projectjava01/src/main/java/libdiaoyong.so");
  4. }
  5.  
  6. public static int getResult() {
  7. return 2;
  8. }
  9.  
  10. public static native int get();
  11.  
  12. public static void main(String[] args) {
  13. TestJNI.get();
  14. }
  15. }

如上实例在main()方法中调用native方法get(),具体的调用过程在前一篇文章中已经详细介绍过。我们这个实例将介绍native方法get()调用Java方法getResult()。

调用的native方法get()的本地函数的实现如下:

  1. JNIEXPORT jint JNICALL Java_TestJNI_get(JNIEnv * env, jclass jc){
  2. jclass cls = (*env)->FindClass(env, "TestJNI");
  3. jmethodID id = (*env)->GetStaticMethodID(env, jc, "getResult", "()I");
  4. jint x = (*env)->CallStaticIntMethod(env,cls, id);
  5. return x;
  6. }

在如上的函数中,我们会调用TestJNI类的getResult()方法获取一个整数值,然后本地函数返回这个整数值。

调用的CallStaticIntMethod()函数的实现如下:

  1. extern "C" {
  2. jint jni_CallStaticIntMethod(JNIEnv *env, jclass cls, jmethodID methodID, ...) {
  3. JavaThread* thread=JavaThread::thread_from_jni_environment(env);
  4. ThreadInVMfromNative __tiv(thread);
  5.  
  6. HandleMarkCleaner __hm(thread);
  7. Thread* __the_thread__ = thread;
  8. os::verify_stack_alignment();
  9. WeakPreserveExceptionMark __wem(thread);
  10. jint ret = 0;
  11.  
  12. va_list args;
  13. __builtin_va_start(args,methodID);
  14. JavaValue jvalue(T_INT);
  15. JNI_ArgumentPusherVaArg ap(methodID, args);
  16. jni_invoke_static(env, &jvalue, 0, JNI_STATIC, methodID, &ap, __the_thread__);
  17.  
  18. // ...
  19.  
  20. __builtin_va_end(args);
  21. ret = jvalue.get_jint();
  22. return ret;
  23. }
  24. }

调用的jni_invoke_static()函数的实现如下:

  1. // 通过JNI的方式调用Java静态方法
  2. static void jni_invoke_static(
  3. JNIEnv *env,
  4. JavaValue* result,
  5. jobject receiver,
  6. JNICallType call_type,
  7. jmethodID method_id,
  8. JNI_ArgumentPusher *args,
  9. TRAPS
  10. ){
  11. Method* m = Method::resolve_jmethod_id(method_id);
  12. methodHandle method(THREAD, m);
  13.  
  14. ResourceMark rm(THREAD);
  15. int number_of_parameters = method->size_of_parameters();
  16. // 这里将要传给Java方法的参数转换为JavaCallArguments实例传下去
  17. JavaCallArguments java_args(number_of_parameters);
  18. args->set_java_argument_object(&java_args);
  19.  
  20. // 通过方法指纹值填写JavaCallArguments实例
  21. Fingerprinter fp = Fingerprinter(method);
  22. uint64_t x = fp.fingerprint();
  23. args->iterate(x);
  24. // 初始化方法返回类型
  25. BasicType bt = args->get_ret_type();
  26. result->set_type(bt);
  27.  
  28. // 调用java方法
  29. JavaCalls::call(result, method, &java_args, CHECK);
  30.  
  31. // 当Java方法返回对象类型数据时,需要句柄化后存储到result中
  32. if (result->get_type() == T_OBJECT || result->get_type() == T_ARRAY) {
  33. oop tmp = (oop) result->get_jobject();
  34. jobject jobj = JNIHandles::make_local(env,tmp);
  35. result->set_jobject(jobj);
  36. }
  37. }

调用的JavaCalls::call()函数最终会调用到JavaCalls::call_helper()函数,这个函数在之前介绍解释执行Java方法时详细介绍过。当调用到Java方法getResult()时,栈的状态如下图所示。

其中的蓝色为C/C++函数的栈帧,而绿色为Java方法的栈帧。

现在我们要从解释执行的main()方法调用native方法get(),这在前一篇详细介绍过,而从Java_TestJNI_get()函数调用getResult()的过程非常类似于HotSpot VM调用main()方法的过程,关于HotSpot VM调用main()方法的过程在之前详细介绍过,这里不再详细介绍。这里我们介绍一下JavaCallWrapper类。

通过上图我们能够看出,Java栈和C/C++栈混合在一起,这就为不同类型栈的展开(如GC需要遍历栈帧、异常需要向上查找异常处理器等),不同类型栈帧的转换和适配增加了不少难度,这些我们在后面都会详细介绍。

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

第48篇-native方法调用解释执行的Java方法的更多相关文章

  1. 第47篇-解释执行的Java方法调用native方法小实例

    举个小实例,如下: public class TestJNI { static { // 程序在加载时,自动加载libdiaoyong.so库 System.loadLibrary("dia ...

  2. 第44篇-为native方法设置解释执行入口

    对于Java中的native方法来说,实际上调用的是C/C++实现的本地函数,由于可能会在Java解释执行过程中调用native方法,或在本地函数的实现过程中调用Java方法,所以当两者相互调用时,必 ...

  3. 意外作出了一个javascript的服务器,可以通过js调用并执行任何java(包括 所有java 内核基本库)及C#类库,并最终由 C# 执行你提交的javascript代码! 不敢藏私,特与大家分

    最近研发BDC 云开发部署平台的数据路由及服务管理器意外作出了一个javascript的服务器,可以通过js调用并执行任何java(包括 所有java 内核基本库)及C#类库,并最终由 C# 执行你提 ...

  4. static 关键字详解 static方法调用非static属性和方法

    静态的属性和方法在内存中的存放地址与非静态的是不同的,静态的是存放在static区,它意味着静态方法是没有this的,所以我们不可以从一个static方法内部发出对非static方法的调用.但是反之是 ...

  5. 生命周期方法调用,以及在onStop()方法中处理草稿信息

    生命周期方法调用顺序 1. 从会话列表界面跳转到信息列表界面. 07-17 17:29:18.718: I/txrjsms(19370): MessageListActivity.onCreate 0 ...

  6. Java方法调用机制

    最近在编程时,修改方法传入对象的对象引用,并没有将修改反映到调用方法中.奇怪为什么结果没有变化,原因是遗忘了Java对象引用和内存分配机制.本文介绍3个点: ① 该问题举例说明 ② 简要阐述Java内 ...

  7. 04 JVM是如何执行方法调用的(下)

    虚方法调用 Java 里所有非私有实例方法调用都会被编译成 invokevirtual 指令,而接口方法调用会被编译成 invokeinterface 指令.这两种指令,均属于 Java 虚拟机中的虚 ...

  8. 第5篇-调用Java方法后弹出栈帧及处理返回结果

    在前一篇 第4篇-JVM终于开始调用Java主类的main()方法啦 介绍了通过callq调用entry point,不过我们并没有看完generate_call_stub()函数的实现.接下来在ge ...

  9. Struts2学习笔记 - Action篇<动态方法调用>

    有三种方法可以使一个Action处理多个请求 动态方法调用DMI 定义逻辑Acton 在配置文件中使用通配符 这里就说一下Dynamic Method nvocation ,动态方法调用,什么是动态方 ...

随机推荐

  1. HDFS04 HDFS的读写流程

    HDFS的读写流程(面试重点) 目录 HDFS的读写流程(面试重点) HDFS写数据流程 网络拓扑-节点距离计算 机架感知(副本存储节点的选择) HDFS的读数据流程 HDFS写数据流程 客服端把D: ...

  2. volatile原理和应用场景

    volatile是java语言中的一个关键字,常用于并发编程,有两个重要的特点:具有可见性,java虚拟机实现会为其满足Happens before原则;不具备原子性.用法是修饰变量,如:volati ...

  3. react动态添加样式:style和className

    react开发过程中,经常会需要动态向元素内添加样式style或className,那么应该如何动态添加呢??? 一.react向元素内,动态添加style 例如:有一个DIV元素, 需要动态添加一个 ...

  4. Gradle插件详解

    参考[1]Gradle 插件       [2]修改 Gradle 插件(Plugins)的下载地址(repositories)

  5. ORACLE 按逗号拆分字符串为多行

    with t as (select '1,2,3,10,11,12' a from dual) select substr(a, decode(level - 1, 0, 0, instr(a, ', ...

  6. javaAPI2

    ---------------------------------------------------------------------------------------------------- ...

  7. Spring Boot发布war包流程

    1.修改web model的pom.xml <packaging>war</packaging> SpringBoot默认发布的都是jar,因此要修改默认的打包方式jar为wa ...

  8. Vue中如何书写js来渲染页面填充数据的部分代码

    new Vue({ el:"#app" , data:{ user:{ id:"", username:"", password:" ...

  9. CPU的中断

    目录 一.简介 二.具体 方式 硬中断 软中断 中断切换 网卡中断 三.中断查看 一.简介 中断其实就是由硬件或软件所发送的一种称为IRQ(中断请求)的信号.中断允许让设备,如键盘,串口卡,并口等设备 ...

  10. [BUUCTF]PWN——ez_pz_hackover_2016

    ez_pz_hackover_2016 题目附件 解题步骤: 例行检查,32位,开启了RELRO保护,二进制的保护机制看这里 由于没有开启nx保护,对于这题一开始想到的是利用写入shellcode来获 ...