传智播客JNI第七讲 – JNI中的全局引用/局部引用/弱全局引用、缓存jfieldID和jmethodID的两种方式

 讲解JNI中的全局引用/局部引用/弱全局引用、缓存jfieldID和jmethodID的两种方式,并编写两种缓存方式的示例代码。java

1.从Java虚拟机建立的对象传到本地C/C++代码时会产生引用,根据Java的垃圾回收机制,只要有引用存在就不会出发该引用指向的Java对象的垃圾回收。c++

2.这些引用在JNI中分为三种:
  全局引用:Global Reference
  局部引用:Local Reference
  若全局引用:Weak Global Reference since JDK1.2编程

3.局部引用
  1)最多见的引用类型,基本上经过JNI返回来的引用都是局部引用。例如使用NewObject就会返回建立出来的实例的局部引用,局部引用只在该native函数中有效,全部在该函数中产生的局部引用,都会在函数返回的时候自动释放,也可使用DeleteLocalRef函数手动释放该引用。
  2)想想既然局部引用可以在函数返回时自动释放,为何还须要DeleteLocalRef函数呢?
  3)实际上,局部引用存在,就会防止其指向的对象被垃圾回收,尤为是当一个局部引用指向一个很庞大的对象,或是在一个循环中生成了局部引用,最好的作法就是在使用完该对象后,或在循环尾部把这个引用释放掉,以确保在垃圾回收器被处罚的时候被回收。
  4)在局部引用的有效期中,能够传递到别的本地函数中,要强调的是他的有效期仍然只在一次的Java本地函数调用中,因此千万不能用C++全局变量保存它或者把它定义为C++静态局部变量。数组

4.全局引用
  1)全局引用能够跨越当前线程,在多个native函数中有效,不过须要编程人员手动来释放该引用,全局引用存在期间会防止在Java的垃圾回收。
  2)与局部引用不一样,全局引用的建立不是由JNI自动建立的,全局引用时须要调用NewGlobalRef函数,而释放它须要使用ReleaseGlobalRef函数。缓存

5.弱全局引用
  1)Java1.2新出来的功能,与全局引用类似,建立跟删除都须要由编程人员来进行。这种引用与全局引用同样能够再多个本地代码有效,也跨越多线程有效,不同的是,这种引用将不会阻止垃圾回收器回收这个引用所指向的对象。
  2)使用NewWeakGlobalRef跟ReleaseWeakGlobalRef来产生和解除引用。多线程

6.关于引用的一些函数
  jobject NewGlobalRef(jobject obj);
  jobject NewLocalRef(jobject obj);
  jobject NewWeakGlobalRef(jobject obj);
  void DeleteGlobalRef(jobject obj);
  void DeleteLocalRef(jobject obj);
  void DeleteWeakGlobalRef(jobject obj);
  jboolean IsSameObject(jobject obj1, jobject obj2); // 这个函数对于弱全局引用还有一个特别的功能,把NULL传入要比较的对象中,就可以判断弱全局引用所指向的Java对象是否被回收。ide

7.缓存jfieldID,jmethodID
  1)取得jieldID跟jmethodID的时候会经过该属性、方法名称加上签名来查询相应的jfieldID,jmethodID。这种查询相对来讲开销较大,咱们能够将这些FieldID,MethodID缓存起来,这样只须要查询一次,之后就使用缓存起来的FieldID,MethodID。
  2)介绍两种缓存方式
   1.在用的时候缓存 
   2.在Java类初始化时缓存函数

  11)在第一次使用的时候缓存
     在native code中使用static局部变量来保存已经查询过的id,这样就不会再每次的函数调用时查询,而只要第一次查询成功后就保存起来了。
     不过在这种状况下就不得不考虑多线程同时呼叫此函数时可能会招致同时查询的危机,不过这种状况是无害的,由于查询同一个属性,方法的ID一般返回的是同样的值。
     JNIEXPORT void JNICALL Java_Test_native(JNIEnv* env, jobject obj){
 static jfieldID fieldID_string = NULL;
 jclass clazz = env->GEtObjectClass(obj);
 if(fieldID_string == NULL){
    fieldID_string = env->GetFieldID(clazz, "string", "Ljava/lang/String;");
 }
 // other code...学习

       }
 
     22)在Java类初始化的时候缓存
        更好的一个方式就是在任何native函数调用前把id所有存起来。
 咱们可让Java在第一次加载这个类的时候首先调用本地代码初始化全部的jfieldID,jmethodID,这样的话,就能够省去屡次的肯定id是否存在的语句,固然,这些jfieldID,jmethodID是定义在C/C++的全局。
 使用这种方式的好处,当Java类卸载或是从新加载的时候,也会从新呼叫该本地代码来从新计算IDs。ui

 

 

 

课程最后总结
在这一课中,咱们学习了:
1.最简单的Java调用C/C++函数的方法
2.取得方法、属性的ID,学会了取得/设置属性,还有Java函数的调用。
3.Java/c++之间的字符串的转换问题。
4.在C/C++下如何操做Java数组。
5.三种引用方式
6.如何缓存属性和方法的ID

使用JNI的两个弊端
1.使用了JNI,那么这个应用就不能跨平台了,若是须要移植到别的平台上,那么native代码就须要从新编写。
2.Java是强类型的语言,而C/C++不是,你必须写JNI时更当心。
3.尽可能少使用本地代码。

其它 1.异常处理 2.C/C++如何启动JVM 3.JNI跟多线程     介绍两本书做为参考:   1)The Java Native Interface Programmer's Guide and Specification   2))JNI++ User Guide

相关文章
相关标签/搜索