JVM 本地方法栈(native method stack)解释

本文详细解析了Java虚拟机中的本地方法栈概念及其工作原理,介绍了Native方法的作用及其实现方式,通过示例代码展示了Native方法的最小实现过程。

本文内容如有错误、不足之处,欢迎技术爱好者们一同探讨,在本文下面讨论区留言,感谢。

什么是本地方法栈?

oracle官方文档jvms-se7

Java虚拟机的实现可以使用传统的堆栈(俗称“ C堆栈”)来支持native方法(用Java编程语言以外的语言编写的方法)。解释器的实现也可以使用诸如C之类的语言来解释Java虚拟机的指令集,以使用native 本地方法栈。无法加载方法并且自身不依赖于常规堆栈的Java虚拟机实现无需提供本地方法栈。如果提供,通常在创建每个线程时为每个线程分配本地方法栈。

该规范允许本地方法堆栈具有固定大小,或者根据计算要求动态扩展和收缩。如果本地方法栈的大小固定,则在创建每个本地方法栈的大小时可以独立选择。

Java虚拟机实现可以为程序员或用户提供对本地方法栈的初始大小的控制,并且在本地方法堆栈大小变化的情况下,可以控制最大和最小方法栈大小。

以下异常条件与本方法堆栈相关联:

  1. 如果线程中的计算所需的本地方法栈超出允许的范围,则Java虚拟机将抛出StackOverflowError。
  2. 如果可以动态扩展本法方法栈并尝试进行本法方法栈扩展,但是可以提供足够的内存,或者可以提供足够的内存来为新线程创建初始本法方法栈,则Java虚拟机将抛出OutOfMemoryError。
native方法是什么?

参考资料:

  • https://stackoverflow.com/questions/6101311/what-is-the-native-keyword-in-java-for
  • https://stackoverflow.com/questions/18824798/what-is-difference-between-java-method-and-native-method
作用

native方法 也叫做本地方法。

  1. native 是Java中的关键字,表示平台相关,用于调用本地代码。
  2. native方法充当通过JNI(Java本地接口)或JNA(Java本地访问)链接到其他编程语言(sun使用的是C/C++语言)之间的接口。比如:public native void method();

本地方法是以非Java语言开始的Java方法(Java方法是只提供方法声明,非Java语言提供具体实现)。本地方法可以访问特定于系统的功能和API,而这些功能和API在Java中不直接可用(比如Unsafe类中的native方法,Unsafe只能通过反射进行创建)。

下面介绍stackoverflow上的一个最小实现来帮助理解上面的概念。

native 最小的实现

Main.java

public class Main {
    public native int square(int i);
    public static void main(String[] args) {
        System.loadLibrary("Main");
        System.out.println(new Main().square(2));
    }
}

Main.c

#include <jni.h>
#include "Main.h"

JNIEXPORT jint JNICALL Java_Main_square(
    JNIEnv *env, jclass obj, jint i) {
  return i * i;
}

编译并运行:

sudo apt-get install build-essential openjdk-7-jdk
export JAVA_HOME='/usr/lib/jvm/java-7-openjdk-amd64'
javac Main.java
javah -jni Main
gcc -shared -fpic -o libMain.so -I${JAVA_HOME}/include \
  -I${JAVA_HOME}/include/linux Main.c
java -Djava.library.path=. Main

输出:

4

上面是在Ubuntu 14.04 AMD64上测试。使用的是Oracle JDK 1.8.0_45

GitHub上的示例供您使用。

Java包/文件名中的下划线必须使用_1进行转义成C方法,如下所述:在包含下划线的Android包名中调用JNI函数

解释

动作:

  • 使用Java中的任意汇编代码调用经过编译的动态加载的库(此处用C编写)
  • 并将结果返回Java

作用:

  • 使用更好的CPU组装指令(不是CPU可移植的)在关键部分上编写更快的代码
  • 进行直接系统调用

这样做是以降低便携性为代价。当然也可以从C调用Java,但是必须首先在C中创建JVM:如何从C ++调用Java函数?

Android NDK

除了必须使用Android样板进行设置外,此概念在此情况下完全相同。

官方的NDK存储库包含“规范”示例,例如hello-jni应用程序:

  • https://github.com/googlesamples/android-ndk/blob/4df5a2705e471a0818c6b2dbc26b8e315d89d307/hello-jni/app/src/main/java/com/example/hellojni/HelloJni.java#L39
  • https://github.com/googlesamples/android-ndk/blob/4df5a2705e471a0818c6b2dbc26b8e315d89d307/hello-jni/app/src/main/cpp/hello-jni.c#L27

此外,file/data/app/com.android.appname-*/oat/arm64/base.odex表示这是一个共享库,我认为这是与ART中的Java文件相对应的AOT预编译.dex,另请参见:Android中的ODEX文件是什么?那么也许Java实际上也可以通过native接口运行?

OpenJDK 8中的示例

首先找到Object#clonejdk8u60-b27中定义的位置,并得出结论,它是通过native调用实现的。

发现

find . -name Object.java

jdk/src/share/classes/java/lang/Object.java#l212

protected native Object clone() throws CloneNotSupportedException;

现在来了困难的部分,在所有间接寻址中找到克隆的位置。使用查询:find . -iname object.c
它将找到可能实现Object的本地方法的C或C ++文件。

jdk/share/native/java/lang/Object.c#l47

static JNINativeMethod methods[] = {
    ...
    {"clone",       "()Ljava/lang/Object;",   (void *)&JVM_Clone},
};

JNIEXPORT void JNICALL
Java_java_lang_Object_registerNatives(JNIEnv *env, jclass cls)
{
    (*env)->RegisterNatives(env, cls,
                            methods, sizeof(methods)/sizeof(methods[0]));
}

找下JVM_Clone符号:

grep -R JVM_Clone

进入hotspot/src/share/vm/prims/jvm.cpp#l580

JVM_ENTRY(jobject, JVM_Clone(JNIEnv* env, jobject handle))
    JVMWrapper("JVM_Clone");

扩展了一堆宏之后,得出的结论是这是定义点。

### JVM本地方法的作用和工作原理 #### 1. 本地方法的定义与作用 JVM中的本地方法Native Method Stack)是一个专门用于管理本地方法调用的数据结构。它主要用于存储由Java代码调用的本地方法的相关信息,例如参数、返回值以及局部变量等[^2]。 当一个Java程序通过`native`关键字调用了某个本地方法时,JVM会在本地方法中创建一个新的帧(Frame),用来保存该方法执行期间所需的信息。 本地方法的主要作用可以概括如下: - **支持跨语言调用**:允许Java代码调用其他编程语言编写的函数或库文件,从而扩展了Java的功能范围。 - **提供底层资源访问能力**:对于需要直接操作硬件设备或者操作系统特性的场景,可以通过本地方法间接完成此类任务。 - **提升性能**:某些计算密集型的操作可能更适合用C/C++实现,在这种情况下利用本地方法能够显著改善应用的整体表现[^4]。 #### 2. 工作原理详解 以下是关于如何运作的一个更深入探讨: ##### (a) 调度过程 每当遇到带有`native`修饰符的方法被触发时,控制权就会转移到相应的原生实现部分上去;此时,当前线程上的虚拟机会暂停其常规处理流程,并切换至对应的本机环境之中继续往下走直到结束为止再回来恢复原来的状态[^1]。 具体来说就是——一旦进入到了任何一个标记有此属性的过程内部之后,则意味着即将离开纯JAVA领域而进入到另一个完全不同的世界里去了(即所谓的“宿主机”) ,在那里按照预先设定好的规则去做一些事情然后再回到起点处等待进一步指示... ##### (b) 数据传递机制 在实际开发过程中,经常可以看到这样的情况发生:我们需要把来自高层级抽象层面上的对象实例或者其他形式的数据传送给低级别的组件去加以运用 。那么这里就涉及到一个问题 —— 如何安全有效地跨越边界呢?这就需要用到JNI技术啦! 简单来讲就是借助于特定格式编码后的句柄作为中介桥梁连接起来两端之间相互沟通交流所需的全部要素 :一方面可以把复杂类型的实体转换成简单的原始数值类型以便传输过去供那边解析重构出来使用;另一方面也能够让对方产生的结果反馈给我们这边重新组装还原回去形成最终想要得到的东西[^4]. ```c JNIEXPORT void JNICALL Java_NativeDemo_printSystemInfo(JNIEnv *env, jobject obj){ printf("This is a native function.\n"); } ``` 上面展示了一个典型的例子展示了如何去编写一段最基础版本的小插件模块来响应外部请求调用自己所提供的服务功能. #### 3. 特殊注意点 尽管存在诸多优点但是同时也伴随着一定风险需要注意防范规避潜在隐患比如内存泄漏等问题可能会由于不当管理造成难以预料后果所以建议开发者们谨慎对待合理规划布局整个项目架构体系确保稳定可靠高效运行[^3]。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值