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. C#中的类、方法和属性

    这节讲C#中的类,方法,属性.这是编码中我们最直接打交道的三个结构.      类: 类(class)是面向对象中最基本的单元,它是一种抽象,对现实世界中事物的抽象,在C#中使用class关键字声明一 ...

  2. 【敏杰开发】Beta阶段事后分析

    [敏杰开发]Beta阶段事后分析 设想和目标 Q 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? 我们达到目标了么(原计划的功能做到了几个? 按照原计划交付时间交付 ...

  3. 持续集成和持续交付工具-jenkins

    jenkins说明 jenkins是一款由Java编写的开源的持续集成工具,它运行在Servlet容器中(例如Apache Tomcat).它支持软件配置管理(SCM)工具(包括AccuRev SCM ...

  4. [bug] HDFS:DataXceiver error processing WRITE_BLOCK operation

    文件格式有误,导致读取错误,我的是把制表符敲成了空格

  5. S3待机 S4休眠

    https://hceng.cn/2018/01/18/Linux%E7%94%B5%E6%BA%90%E7%AE%A1%E7%90%86/ 1.1系统睡眠模型Suspend On (on) S0 - ...

  6. docker 日志位置

    日志分两类,一类是 Docker 引擎日志:另一类是 容器日志. Docker 引擎日志 Docker 引擎日志 一般是交给了 Upstart(Ubuntu 14.04) 或者 systemd (Ce ...

  7. Linux创建RAID5_实战

    Linux创建RAID5实战 Linux创建RAID5 RAID5最少由三个硬盘组成,它将数据分散存储于阵列中的每个硬盘,并且还伴有一个数据校验位,数据位与校验位通过算法能相互验证 RAID5最多能允 ...

  8. IDEA 全局搜索 Jar 包中源码内容

    引言 项目开发过程中,经常遇到需要在依赖的 Jar 包查看源码,查找类方法和属性,介绍两种在 IDEA 中搜索 Jar 包内容的方式 方式一:双击 SHIFT 快捷键 输入需要查询的类名或方法名 方式 ...

  9. 安装oracle 时“[INS-30014]无法检查指定的位置是否位于 CFS上”问题

    错误截图: 错误信息: [INS-30014]无法检查指定的位置是否位于 CFS上 解决方案: 通过修改hosts文件,向C:\Windows\System32\drivers\etc\hosts文件 ...

  10. 安装T4环境

    Install-Package Microsoft.VisualStudio.TextTemplating.14.0 -Version 14.3.25407