本文转载整理自:
http://my.unix-center.net/~Simon_fu/?p=849
http://docs.oracle.com/javase/1.5.0/docs/guide/jni/spec/functions.html
在JDK1.2中为了方便管理局部引用,引入了三个函数——EnsureLocalCapacity、PushLocalFrame、PopLocalFrame。这里介绍一下PushLocalFrame和PopLocalFrame函数。这两个函数是成对使用的,先调用PushLocalFrame,然后创建局部引用,并对其进行处理,最后调用PushLocalFrame释放局部引用,这时Java虚拟机也可以对其指向的对象进行垃圾回收。可以用C语言的栈来理解这对JNI API,调用PushLocalFrame之后Native代码创建的所有局部引用全部入栈,当调用PopLocalFrame之后,入栈的局部引用除了需要返回的局部引用(PushLocalFrame和PopLocalFrame这对函数可以返回一个局部引用给外部)之外,全部出栈,Java虚拟机这时可以释放他们指向的对象。具体的用法可以参考手册。这两个函数使JNI的局部引用由于和C语言的局部变量用法类似,所以强烈推荐使用。
一、简介
JNI规范中定义了三种引用——全局引用(Global reference),局部引用(Local reference),弱全局引用(Weak global reference)。
这算三种引用的生存期是不同的。
全局引用的生存期为创建之后,直到程序员显式的释放它。
局部引用的生存期为创建后,直到程序员显式的释放他们,或在当前上下文(可以理解成Java程序调用Native代码的过程)结束之后没有被JVM发现有JAVA层引用而被JVM回收并释放。
弱全局引用的生存期为创建之后,到程序员显式的释放他们或JVM认为应该回收它的时候(比如内存紧张的时候)进行回收而被释放。
二、局部引用
这里重点要强调一下局部引用的有效期,很多有C语言背景的程序员会认为当Native函数结束之后局部引用就无效了,和C语言的局部变量对应。实际上JNI中的局部引用和C语言中局部变量是不同的,它的有效期不只是当前Native函数被调用的上下文中。我理解的调用上下文,为Java虚拟机的调用流程。Native函数是被Java虚拟机调用的,Native函数执行完成之后,控制流程将继续返回给Java虚拟机。局部变量在Native函数中,由Native代码调用Java虚拟机的JNI接口创建,秉着谁创建谁销毁的原则(软件设计一个常用规则),当Native函数执行完成之后,如果局部引用没有被Native代码显式删除,那么局部引用在Java虚拟机中还是有效的。Java虚拟机来决定在什么时候来删除这个对象,而且直到JAVA层没有对它的引用(可以通过Native函数返回而把它引用到JAVA层),它才能被JVM回收并释放。这和C语言的局部变量概念是不同的。这也可以解释为什么Natvie函数能够以一个局部引用为返回值了。
局部引用在Native代码显示释放非常重要。你可能会问,既然Java虚拟机会自动释放局部变量为什么还需要我在Native代码中显示释放呢?原因有以下几点:
1、Java虚拟机默认为Native引用分配的局部引用数量是有限的,大部分的Java虚拟机实现默认分配16个局部引用。当然Java虚拟机也提供API(PushLocalFrame,EnsureLocalCapacity)让你申请更多的局部引用数量(Java虚拟机不保证你一定能申请到)。有限的资源当然要省着点用,否则将会被Java虚拟机无情抛弃(程序崩溃)。JNI编程中,实现Native代码时强烈建议调用PushLocalFrame,EnsureLocalCapacity来确保Java虚拟机为你准备好了局部变量空间。
2、如果你实现的Native函数是工具函数,会被频繁的调用。如果你在Native函数中没有显示删除局部引用,那么每次调用该函数Java虚拟机都会创建一个新的局部引用,造成局部引用过多。尤其是该函数在Native代码中被频繁调用,代码的控制权没有交还给Java虚拟机,所以Java虚拟机根本没有机会释放这些局部变量。退一步讲,就算该函数直接返回给Java虚拟机,也不能保证没有问题,我们不能假设Native函数返回Java虚拟机之后,Java虚拟机马上就会回收Native函数中创建的局部引用,依赖于Java虚拟机实现。所以我们在实现Native函数时一定要记着删除不必要的局部引用,否则你的程序就有潜在的风险,不知道什么时候就会爆发。
3、如果你Native函数根本就不返回。比如消息循环函数——死循环等待消息,处理消息。如果你不显示删除局部引用,很快将会造成Java虚拟机的局部引用内存溢出。
在JNI中显示释放局部引用的函数为DeleteLocalRef,大家可以查看手册来了解调用方法。
当创建局部变量之后,Java虚拟机直到Native代码显示调用了DeleteLocalRef删除局部引用或从Native返回且没有另外的引用才能对该对象进行回收。Native代码调用DeleteLocalRef显示删除局部引用之后,Java虚拟机就可以对局部引用指向的对象垃圾回收了。当Native代码创建了局部引用,但未显示调用DeleteLocalRef删除局部引用,并返回Java虚拟机的话,那么由虚拟机来决定什么时候删除该局部引用,然后对其指向的对象垃圾回收。程序员不能对java虚拟机删除局部引用的时机进行假设。
局部引用仅仅对于java虚拟机当前调用上下文有效,不能够在多次调用上下文中共享局部引用。这句话也可以这样理解:局部引用只对当前线程有效,多个线程之间不能共享局部引用。局部引用不能用C语言的静态变量或者全局变量来保存,否则第二次调用的时候,将会产生崩溃。
三、全局引用
在所有引用中,我觉得全局引用是最好理解的一个了。为什么呢?主要和C语言的全局变量非常相近。
我已经提到局部引用大部分是通过JNI API返回而创建的,而全局引用必须要在Native代码中显示的调用JNI API NewGlobalRef来创建,创建之后将一直有效,直到显示的调用DeleteGlobalRef来删除这个全局引用。请注意NewGlobalRef的第二个参数,既可以用一个局部引用,也可以用全局引用生成一个全局引用,当然也可以用弱全局引用生成一个全局引用,但是这中情况有特殊的用途,后文会介绍。
全局引用和局部引用一样,可以防止其指向的对象被Java虚拟机垃圾回收。与局部引用只在当前线程有效不同的是全局引用可以在多线程之间共享(如果是多线程编程需要注意同步问题)。
四、弱全局引用
弱全局引用和全局引用一样,可以在多个线程之间共享,但是并不强制进行显式的销毁。虽然在我们确定不再需要弱全局引用的时候,建议进行显式的销毁(调用DeleteWeakGlobalRef)。但是即使我们不显式的销毁弱全局引用,JAVA虚拟机也能在它认为必要的时候自动回收并销毁弱全局引用。创建弱全局引用请使用NewWeakGlobalRef,显式销毁弱全局引用请使用DeleteWeakGlobalRef。
与全局引用和局部引用能够阻止Java虚拟机垃圾回收其指向的对象不同,弱全局引用指向的对象随时都可以被Java虚拟机垃圾回收,所以使用弱全局变量的时候,要时刻记着:它所指向的对象可能已经被垃圾回收了。JNI API提供了引用比较函数IsSameObject,用弱全局引用和NULL进行比较,如果返回JNI_TRUE,则说明弱全局引用指向的对象已经被释放。需要重新初始化弱全局引用。根据上面的介绍你可能会写出如下的代码:
static jobject weak_global_ref = NULL;
if((*env)->IsSameObject(env, weak_global_ref, NULL) == JNI_TRUE)
{
/* Init week global referrence again */
weak_global_ref = NewWeakGlobalRef(...);
}
/* Process weak_global_ref */
上面这段代码表面上没有什么错误,但是我们忘了一点儿,Java虚拟机的垃圾回收随时都可能发生。假设如下情形:
1、通过引用比较函数IsSameObject判断弱全局引用是否有效的时候,返回JNI_FALSE,证明其指向对象有效。
2、这时Java虚拟机进行了垃圾回收,回收了弱全局引用指向的对象。
3、这样如果我们后面访问弱全局引用指向的对象,将会引发程序崩溃,因为弱全局引用指向对象已经被Java虚拟机回收了。
根据JNI标准手册《Weak Global References》中的介绍,我们可以有这样一个使用弱全局引用的方案。在使用全局引用之前,我们先通过NewLocalRef函数创建一个局部引用,然后使用这个局部引用来访问该对象进行处理,当完成处理之后,删除局部引用。局部引用可以阻止Java虚拟机回收其指向的对象,这样可以保证在处理期间弱全局引用和局部引用指向的对象不会被Java虚拟机回收。假如弱全局引用指向对象已经被Java虚拟机回收,则NewLocalRef函数将会返回NULL,则创建局部引用失败,这个返回值有助于我们判断是否需要重新初始化弱全局引用。我们可以写出如下的代码:
static jobject weak_global_ref = NULL;
jobject local_ref;
/* We ensure create local_ref success */
while ( week_global_ref == NULL
|| (local_ref = NewLocalRef(env, weak_global_ref)) == NULL )
{
/* Init week global referrence again */
weak_global_ref = NewWeakGlobalRef(...);
}
/* Process local_ref */
.....
(*env)->DeleteLocalRef(env, local_ref);
注意在《Java Native Interface: Programmer’s Guide and Specification》的例子中,有很多不是按照如上的代码实现的,那些代码是有潜在风险的,请各位朋友注意。
弱全局引用是可以用来缓存jclass对象,但是用全局引用来缓存jclass对象将非常的危险。这里需要简单介绍一下Native的共享库的卸载。当Class Loader释放完所有的class后,然后Class Loader会卸载Native的共享库。如果我们用全局引用来缓存jclass对象的话,根据前面对全局引用对Java虚拟机垃圾回收机制的影响,将会阻止Java虚拟机回收该对象。如果我们不显式的释放全局引用(通过DeleteGlobalRef),则Class Loader也将不能释放这个jclass对象,进而造成Class Loader不能卸载Native的共享库(永远无法释放)。如果用弱全局引用来缓存将不会有这个问题,Java虚拟机随时都可以释放它指向的对象。
五、总结
至此我们把JNI规范中的三种引用都进行了一个简单的介绍,在此我对以上内容做一个简单总结:
1、局部引用是Native代码中最常用的引用。大部分局部引用都是通过JNI API返回来创建,也可以通过调用NewLocalRef来创建。另外强烈建议Native函数返回值为局部引用。局部引用只在当前调用上下文中有效,所以局部引用不能用Native代码中的静态变量和全局变量来保存。另外时刻要记着Java虚拟机局部引用的个数是有限的,编程的时候强烈建议调用EnsureLocalCapacity,PushLocalFrame 和PopLocalFrame来确保Native代码能够获得足够的局部引用数量。
2、全局变量必须要通过NewGlobalRef创建,通过DeleteGlobalRef删除。主要用来缓存Field ID和Method ID。全局引用可以在多线程之间共享其指向的对象。在C语言中以静态变量和全局变量来保存。
3、全局引用和局部引用可以阻止Java虚拟机回收其指向的对象。
4、弱全局引用必须要通过NewWeakGlobalRef创建,通过DeleteWeakGlobalRef销毁。可以在多线程之间共享其指向的对象。在C语言中通过静态变量和全局变量来保持弱全局引用。弱全局引用指向的对象随时都可能会被Java虚拟机回收,所以使用的时候需要时刻注意检查其有效性。弱全局引用经常用来缓存jclass对象。
5、全局引用和弱全局引用可以在多线程中共享其指向对象,但是在多线程编程中需要注意多线程同步。强烈建议在JNI_OnLoad初始化全局引用和弱全局引用,然后在多线程中进行读全局引用和弱全局引用,这样不需要对全局引用和弱全局引用同步(只有读操作不会出现不一致情况)。
转自:http://hubingforever.blog.163.com/blog/static/171040579201221553444677/
相关推荐
JNI引用类型管理涉及到局部引用、全局引用和弱全局引用,它们各自有不同的生命周期和用途。 **局部引用(Local References)** 局部引用在JNI函数内部创建,与JNI方法调用的生命周期绑定。每当JNI函数被调用,系统...
GC Roots包括虚拟机栈中引用的对象、方法区中的类静态属性引用的对象、常量引用的对象以及本地方法栈中JNI引用的对象等。 三、四种引用状态 在JDK1.2之后,Java对引用的概念进行了扩展,提出了四种引用状态:强...
GC Roots包括虚拟机栈中引用的对象、本地方法栈内的JNI引用、方法区的静态属性和常量引用、同步锁持有的对象等。为了保证分析的准确性,垃圾回收过程需要暂停所有线程,即Stop-The-World(STW)。 3. 标记-清除算法...
### 单例模式与垃圾回收机制 #### 一、引言 在软件开发领域,设计模式作为一种被广泛接受的最佳实践,对于提高代码质量和可维护性起着重要作用。单例模式作为23种经典设计模式之一,确保了某个类只有一个实例,并...
其中,程序计数器、虚拟机栈和本地方法栈与线程生命周期同步,而垃圾回收主要关注的是Java堆和方法区。Java堆用于存储对象实例,方法区则存放类的信息,如类的静态变量和常量池。 在垃圾回收中,主要关注的是对象的...
3. **弱全局引用**:类似于全局引用,但不会阻止对象被垃圾回收。只有当对象被回收后,弱全局引用才失效。使用`NewWeakGlobalRef`创建。 现在回到你的问题,你在JNI中使用了全局引用变量。这可能是为了确保Java对象...
在实际开发中,Delphi程序员需要理解JNI的生命周期管理,比如如何正确地创建和释放本地引用,以防止Java对象过早被垃圾回收。此外,还要注意线程安全问题,因为Java是多线程的,而Delphi的代码可能需要同步控制以...
2. 引用可达性分析:更常见的方法,通过一系列称为“根”(如栈上的局部变量、静态字段、JNI引用等)的对象出发,遍历所有可达的对象,不可达的对象被视为垃圾。这种方法可以有效解决循环引用问题。 二、垃圾收集器...
垃圾回收的主要目标是识别并清理不再被程序引用的对象,以便回收其占用的内存供后续使用。 首先,我们要理解Java内存的几个主要区域。栈内存用于存储方法调用的局部变量和方法参数,每个线程都有自己的独立栈。堆...
一、Java垃圾回收的起源与目的 在C++等语言中,程序员需要手动管理内存,分配和释放内存空间。而在Java中,引入了垃圾回收机制,自动处理这些任务,简化了编程工作,也降低了因忘记释放内存而导致的错误。 二、垃圾...
### JVM工作原理及垃圾回收机制详解 #### 一、JVM概述及原理 **1.1 JVM概述** Java Virtual Machine (JVM),即Java虚拟机,是一种虚构的计算机,在实际的计算机硬件上仿真模拟出的一套完整的计算机系统,用于执行...
**JVM内存基本结构** Java虚拟机(JVM)是Java程序运行的平台,它提供了内存管理、类加载、字节码执行等核心功能。...同时,良好的编程习惯,如及时释放无用对象引用,也能有效减少垃圾回收的压力。
这些引用路径被称为“引用链”,可以从一组特定的根对象开始,如局部变量、活动线程、静态字段、JNI引用和其他系统内部引用。如果一个对象不能从这些根对象通过引用链到达,那么它被认为是不可达的,也就被视为垃圾...
三、JNI引用 JNI提供了三种类型的引用:局部引用、全局引用和弱全局引用。局部引用由JNI函数自动创建,当函数返回时自动释放。全局引用不会自动释放,除非显式调用`DeleteGlobalRef`。弱全局引用与全局引用相似,但...
GC Roots包括虚拟机栈中的本地变量表、方法区中的静态属性和常量引用、本地方法栈中的JNI引用等。 Java中的引用类型有强引用、软引用、弱引用和虚引用。强引用是最常见的引用,对象只要被强引用,就不会被GC回收。...
Java垃圾回收机制是Java编程中的核心概念,它自动管理程序中的内存分配与释放,从而避免了程序员手动管理内存可能导致的内存泄漏等问题。垃圾回收的主要任务是识别并清理不再被程序引用的对象,释放其所占用的内存...
- **概念**: 引用计数算法是一种简单的垃圾回收方式,通过跟踪每个对象被引用的次数来决定其生存状态。 - **工作原理**: 每个对象都有一个引用计数器,每当有引用指向该对象时,计数器增加1;当引用失效或被显式设置...