内存与垃圾回收——(七)本地方法接口

本文探讨了Java本地方法接口(NativeMethod)的概念,解释了其如何允许Java与非Java代码交互,尤其是在操作系统交互和底层系统集成中的应用。重点介绍了使用本地方法的原因,如与外部环境的高效通信,以及与操作系统内核的紧密协作。尽管现代应用倾向于异构通信,本地方法在特定硬件驱动和设备管理场景仍有所用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

7_本地方法接口

7.1_简介

一个Native Method就是一个Java调用非Java代码的接口。

本地方法接口:

该方法的底层实现由非Java语言实现,比如C。这个特征并非java特有,很多其他的编程语言都有这一机制,比如在C++ 中,你可以用extern "C" 告知C++ 编译器去调用一个C的函数。

在定义一个native method时,并不提供实现体(有些像定义一个Java interface),因为其实现体是由非Java语言在外面实现的。

本地接口的作用是融合不同的编程语言为Java所用,它的初衷是融合C/C++程序。

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

7.2_使用Native Method的原因

Java使用起来非常方便,然而有些层次的任务用Java实现起来不容易,或者我们对程序的效率很在意时,问题就来了。

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

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

### JVM 内存模型 Java 虚拟机(JVM内存被划分为多个不同的区域,这些区域各自承担特定的任务并具有独特的特性。 #### 方法区 方法区存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。此部分可以设置参数`-XX:+HeapDumpOnOutOfMemoryError`使得在发生 `OutOfMemoryError` 错误时生成堆转储文件以便分析[^2]。 #### 堆 堆是所有线程共享的一块最大的内存区域,在几乎所有的应用中都是垃圾收集器管理的主要对象。它又细分为新生代和老年代两个子域: - **新生代 (Young Generation)**:大多数情况下新创建的对象都会优先分配在此处。由于这部分经常有大量短生命周期的对象产生,因此频繁执行年轻代回收即Minor GC/Young GC。 - **老年代 (Old Generation)**:经过多次存活下来的对象会被晋升到老年代。这里发生的Full GC涉及整个堆空间,包括永久代或元空间,通常频率较低但耗时较长。 #### Java栈 每个线程都有独立的一个Java栈,它是用来保存局部变量表、操作数栈、动态链接、返回地址等信息的地方。每当启动一条新的线程就会为其创建相应的Java栈实例。 #### 本地方法Java栈作用相似,不过这里是专门为Native Method服务的空间。 #### PC寄存器 程序计数器是一块较小的内存空间,它的唯一目的是记录当前线程所执行字节码指令的位置。 #### 直接内存 虽然不属于传统意义上的JVM运行时数据区组成部分之一,但在某些场景下也会涉及到直接内存的操作。这类内存在使用前需借助于JNI接口向操作系统请求获取,并且不受限于常规意义上由GC负责清理范围之内;如果超出可用限度则可能触发`OutOfMemoryError`异常[^3]。 ### 垃圾回收算法 为了提高效率,针对不同类型的内存分区采用了多种策略相结合的方式来进行垃圾回收工作: #### 新生代回收 对于新生代而言,主要采取的是复制算法。考虑到大部分对象很快便会成为垃圾的特点,只需少量移动存活下来的数据即可完成清理过程。然而这种方式也意味着每次都需要预留足够的备用空间供转移之用。 #### 老年代回收 相比之下,老年代中的对象往往拥有更长的生命期,故而更适合运用标记清除或是标记整理的方法处理。前者能够快速定位待释放的部分,后者除了标注外还会将剩余的有效元素紧凑排列在一起从而减少碎片化现象的发生。 #### 分代回收机制 实际上所谓的分代回收正是基于上述两种思路之上发展起来的一种综合性解决方案——即将整个堆分成若干个小单元分别对待,依据各自的特征选取最合适的清扫手段组合而成。例如,当Eden区或者Survivor区不足以容纳新增加的内容时,则会触发一次轻度的小规模搜集活动(Minor GC),反之若是老年区内出现了容量不足的情况,则不得不进行全面的大扫除(Full GC)[^5]。 ```java // 示例代码展示如何强制建议进行垃圾回收(尽管实际效果取决于JVM实现) public class GCDemo { public static void main(String[] args) { System.gc(); // 提交垃圾回收请求给JVM, 不保证立即执行 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值