JNI相关笔记

1 生成native code所需要的头文件

  • 首先使用javac对java文件进行编译
  • 使用javah -jni [className],生成对应的头文件
  • 创建cpp文件,实现所需要的函数
  • 运行方式:java -Djava.library.path=. HelloWorld或者export LD_LIBRARY_PATH

2 JNI提供的一些函数和方法

  • String对象的获取

    • 获取java传下来的String对象:const jbyte* byteString = (*env)->GetStringUTFChars(env,str,NULL);,由于该函数是对原来字符串的拷贝,所以使用完后需要对byteString进行NULL判断

      使用完毕后,需要释放获取的对象:(*env)->ReleaseStringUTFChars(env,str,byteString);

    • GetStringRegion/GetStringUTFRegion:会将字符串拷贝到预先申请的一片缓冲区里面(在栈上面,因此应避免大对象的拷贝),所以当使用这两个函数的时候,不需要进行NULL判断,也不需要进行Release操作

    • GetStringCritical/ReleaseStringCritical:会尽可能直接使用String的指针(也有可能返回字符串的拷贝),且该方法会阻塞垃圾回收,因此使用的时候需当做临界资源,即在这两个函数中间不能使用JNI调用或者阻塞

  • 类对象的获取

    • 获取当前类的实例:GetObjectClass:jclass cls = (*env)->GetObjectClass(env, obj)

      获取和设置类中成员变量的方法:

        jfieldID fid;
      fid = (*env)->GetFieldID(env, cls, "s", "Ljava/lang/String;");
      //cls为上一步获取的类的实例,GetStaticFieldID获取静态成员变量
      if(!fid) {
      //需要判断是否获取成功
      return; /* failed to find the field !*/
      }
      jstring str = (*env)->GetObjectField(env, obj, fid); //获取jstring对象,获取完后就可以转换为const char*类型来操作了
      jstring target = (*env)->NewStringUTF(env, "123");
      if(!jstr) {
      return; /*out of memory */
      }
      (*env)->SetObjectField(env, obj, fid, jstr); //设置对应的field

    调用java类中的方法:

        jclass cls = (*env)->GetObjectClass(env, obj);
    jmethodID mid = (*env)->GetMethodID(env, cls, "callback", "()V");
    if(!mid) {
    return; /* method not found */
    }
    printf("In C\n");
    //直接调用java对象中的方法,对应的还有CallIntMethod, CallStaticVoidMethod, CallStaticBooleanMethod
    (*env)->CallVoidMethod(env, obj, mid);

3 局部引用,全局引用,全局弱引用。

局部应用的错误示例:https://www.kancloud.cn/owenoranba/jni/120497 以为用static变量来存储findClass的结果,实际上由于findClass返回的局部引用,该引用指向的对象可能已经被销毁,所以这样的做法是无用的

正确的做法是用返回的局部引用创建一个GlobalRef:https://www.kancloud.cn/owenoranba/jni/120498

一般来讲局部引用会自动进行释放,但是这种情况下,最好手动释放局部引用:

for (int i = 0; i < len; ++i) {
jstring jstr = (*env)->GetObjectArrayElement(env, arr, i);
/* ... process jstr */
(*env)->DeleteLocalRef(env, jstr);
}

IsSameObject(jobject obj1,jobject obj2):用来判断两个object是否引用的同一个对象,对于弱全局引用非常有效

EnsureLocalCapacity(jint capacity):确保native函数在调用前有资源能够创建至少capacity个局部引用

(*env)->PushLocalFrame(env, 10)和result = (*env)->PopLocalFrame(env, result):管理局部引用更高效的方式,示例:https://www.kancloud.cn/owenoranba/jni/120505

4 异常

当native代码调用java类中的方法的时候,有时候java类中可能会抛出异常,一般的检测方式为:

(*env)->CallVoidMethod(env, obj, mid);
//ExceptionCheck,更为高效的异常检查方式,返回值为boolean,只关心是否发生异常,不关心异常类的引用
exc = (*env)->ExceptionOccurred(env);
if(exc){
jclass newExcCls;
(*env)->ExceptionDescribe(env);
//打印出异常的描述信息
(*env)->ExceptionClear(env);//需要清理掉异常,为了下一次能够检测到
//省略部分代码
}

对于本地代码来讲,如果需要抛出异常,那么方式为:

jclass newExcCls;
newExcCls = (*env)->FindClass(env, "java/lang/IllegalArgumentException");
if(!newExcCls){
/*Unable to find the exception class, give up. */
return;
} (*env)->ThrowNew(env, newExcCls, "thrown from C code"); //抛出异常一个非常重要的一点,对于native代码来讲,ThrowNew方法并不会中断代码流程,所以需要手动去控制代码流程一个用于方便抛出异常的工具函数:
void JNU_ThrowByName(JNIEnv *env, const char* name, const char* msg){
jclass cls = (*env)->FindClass(env, name);
/*if cls is NULL, an exception has already been thrown */
if(cls){
(*env)->ThrowNew(env, cls, msg);
}
/* free the local ref */
(*env)->DeleteLocalRef(env, cls);//从此处也可以看出来,native代码中的ThrowNew是不影响流程的,需要自行控制
}

JNI相关笔记 [TOC]的更多相关文章

  1. HTTPS证书申请相关笔记

    申请免费的HTTPS证书相关资料 参考资料: HTTPS 检测 苹果ATS检测 什么是ECC证书? 渠道2: Let's Encrypt 优点 缺点 Let's Encrypt 的是否支持非80,44 ...

  2. Android NDK JNI 入门笔记-day04-NDK实现Hash算法

    * Android NDK JNI 入门笔记目录 * 开头 前面的学习,我们已经掌握了 NDK 开发的必备知识. 下一步就要多实践,通过创造问题并解决问题,来增加熟练度,提升经验. 日常开发中,经常会 ...

  3. JNI学习笔记_Java调用C —— Android中使用的方法

    一.笔记 1.JNI(Java Native Interface),就是如何使用java去访问C/C++编写的那些库.若想深入了解JNI可以看官方文档jni.pdf.优秀博文:Android JNI知 ...

  4. JNI 学习笔记

    JNI是Java Native Interface的缩写,JNI是一种机制,有了它就可以在java程序中调用其他native代码,或者使native代码调用java层的代码.也 就是说,有了JNI我们 ...

  5. JNI学习笔记_Java调用C —— 非Android中使用的方法

    一.学习笔记 1.java源码中的JNI函数本机方法声明必须使用native修饰. 2.相对反编译 Java 的 class 字节码文件来说,反汇编.so动态库来分析程序的逻辑要复杂得多,为了应用的安 ...

  6. JNI学习笔记_C调用Java

    一.笔记 1.C调用Java中的方法,参考jni.pdf pg97可以参考博文:http://blog.csdn.net/lhzjj/article/details/26470999步骤: a. 创建 ...

  7. Hadoop相关笔记

    一.            Zookeeper( 分布式协调服务框架 ) 1.    Zookeeper概述和集群搭建: (1)       Zookeeper概述: Zookeeper 是一个分布式 ...

  8. redis相关笔记(二.集群配置及使用)

    redis笔记一 redis笔记二 redis笔记三 1.配置:在原redis-sentinel文件夹中添加{8337,8338,8339,8340}文件夹,且复制原8333中的配置 在上述8333配 ...

  9. redis相关笔记(三.redis设计与实现(笔记))

    redis笔记一 redis笔记二 redis笔记三 1.数据结构 1.1.简单动态字符串: 其属性有int len:长度,int free:空闲长度,char[] bur:字符数组(内容) 获取字符 ...

随机推荐

  1. getInstance()得理解

    使用getInstance()方法的原因及作用 https://www.cnblogs.com/roadone/p/7977544.html 使用getInstance()方法的原因及作用 https ...

  2. 使用JSONassert进行JSON对象对比

      在日常工作中,会接到用户提出一张订单,修改后需要记录每次修改的信息,然后需要查看修改前后的差异信息这样的需求.要实现这样的功能方式有很多.下面介绍下JSONassert的简单使用,也方便自己后续使 ...

  3. 简单对比vue2.x与vue3.x响应式及新功能

    简单对比vue2.x与vue3.x响应式 对响应方式来讲:Vue3.x 将使用Proxy ,取代Vue2.x 版本的 Object.defineProperty. 为何要将Object.defineP ...

  4. jQuery清空元素和克隆元素

    1.清空 $(function () { $('#btn').click(function () { $('#ul1').html('') $('#ul1').empty() $('#ul1').re ...

  5. java集合类介绍

    目录 集合类简介 List ArrayList LinkedList Vector Stack Set HashSet LinkedHashSet TreeSet Map HashMap Hashta ...

  6. [Linux] 删除find到的目录

    参考 https://www.centos.bz/2017/09/linux%E7%B3%BB%E7%BB%9F%E4%B8%8Bfind%E5%91%BD%E4%BB%A4%E9%80%92%E5% ...

  7. [Qt] 基本概念

    QObject :所有 Qt 类的基类 QWidget类:包含所有组件的类 Widgets:组件,组成Qt界面的基本元素 window:界面,是不含有父组件的组件 Child Widgets:子组件, ...

  8. 015.Python函数名的使用以及函数变量的操作

    一 函数名的使用 python中的函数可以像变量一样,动态创建,销毁,当参数传递,作为值返回,叫第一类对象.其他语言功能有限 1.1 函数名是个特殊的变量,可以当做变量赋值 def func(): p ...

  9. cnetos 网卡绑定 eth0+eth1做双网卡绑定到bond0

    1.网卡绑定:eth0+eth1做双网卡绑定到bond0 二.网络配置 网卡绑定1./etc/sysconfig/network-scripts/目录下建立ifcfg-bond0文件,内容如下DEVI ...

  10. Linux进阶之给nginx设置登录用户验证

    一.nginx开启访问验证 使用nginx搭建的站点,如果不想让所有人都能正常访问,那么可以设置访问认证,只有用户输入正确的用户名和密码才能正常访问. 在nginx下,提供了ngx_http_auth ...