[JVM]06.本地方法

1.本地方法接口

1.1什么是本地方法?

简单地讲,一个Native Method就是一个Java调用非Java代码的接口,该方法的实现可以为其他语言。用native关键字修饰的方法,在定义的时候,并不提供实现体,而是由非java语言在外部实现的。本地接口的作用是融合不同的编程语言为Java所用,它的初衷是融合C/C++程序。

标识符native可以与其他所有的java标识符连用,但是abstract除外。

/**
 * 本地方法
 */
public  class IHaveNatives {

    //abstract 没有方法体
    public abstract void abstractMethod(int x);

    //native 和 abstract不能共存,native是有方法体的,由C语言来实现
    public native void Native1(int x);

    native static public long Native2();

    native synchronized private float Native3(Object o);

    native void Native4(int[] array) throws Exception;

}

1.2 为什么要使用本地方法?

Java使用起来非常方便,但是有些层次的任务用Java实现起来非常不容易,或者效率不好。

  • 与Java环境外交互:如与底层系统(操作系统或某些硬件交换信息)。本地方法正是这样一种交流机制:它为我们提供了一个非常简洁的接口,而我们无需去了解Java应用之外的繁琐的细节。
  • 与操作系统交互:JVM支持着java语言本身和运行库,它是java程序赖以生存的平台,它由一个解释器(解释字节码)和一些连接到本地代码的库组成。然而不管怎样,它毕竟不是一个完整的系统,它经常依赖于一些底层系统的支持。这些底层系统常常是强大的操作系统。通过使用本地方法,我们得以用java实现了jre的与底层系统的交互,甚至jvm的一些部分就是用C写的。还有,如果我们要使用一些java语言本身没有提供封装的操作系统特性时,我们也需要使用本地方法。
  • Sun’s Java:Sun的解释器是用C实现的,这使得它能像一些普通的C一样与外部交互。jre大部分是用java实现的,它也通过一些本地方法与外界交互。例如:类java.lang.Thread的setPriority()方法是用Java实现的,但是它实现调用的事该类里的本地方法setPriority0()。这个本地方法是用C实现的,并被植入JVM内部,在Windows 95的平台上,这个本地方法最终将调用Win32 SetProority()API。这是一个本地方法的具体实现由JVM直接提供,更多的情况是本地方法由外部的动态链接库(external dynamic link library)提供,然后被JVM调用。

现状:

目前该方法的是用越来越少了,除非是与硬件有关的应用,比如通过java程序驱动打印机或者java系统管理生产设备,在企业级应用已经比较少见。因为现在的异构领域间的通信很发达,比如可以使用Socket通信,也可以是用Web Service等等,不多做介绍。

2.本地方法栈(Native Method Stack)

Java虚拟机栈用于管理Java方法的调用,而本地方法栈用于管理本地方法的调用

本地方法栈,也是线程私有的。

允许被实现成固定或者是可动态拓展的内存大小。(在内存溢出方面是相同的)

  • 如果线程请求分配的栈容量超过本地方法栈允许的最大容量,Java虚拟机将会抛出一个StackOverFlowError异常。
  • 如果本地方法栈可以动态扩展,并且在尝试扩展的时候无法申请到足够的内存,或者在创建新的线程时没有足够的内存去创建对应的本地方法栈,那么java虚拟机将会抛出一个OutOfMemoryError异常。

本地方法是使用C语言实现的

它的具体做法是Native Method Stack中登记native方法,在Execution Engine执行时加载本地方法库。

在这里插入图片描述

当某个线程调用一个本地方法时,它就进入了一个全新的并且不再受虚拟机限制的世界。它和虚拟机拥有同样的权限

  • 本地方法可以通过本地方法接口来 访问虚拟机内部的运行时数据区
  • 它甚至可以直接使用本地处理器中的寄存器
  • 直接从本地内存的堆中分配任意数量的内存

并不是所有的JVM都支持本地方法。因为Java虚拟机规范并没有明确要求本地方法栈的使用语言、具体实现方式、数据结构等。如果JVM产品不打算支持native方法,也可以无需实现本地方法栈。

在hotSpot JVM中,直接将本地方法栈和虚拟机栈合二为一。

### 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、付费专栏及课程。

余额充值