字符串和数组是JNI中常见的引用数据类型,本文将介绍符串和数组在JNI中的常见处理方式。
JNI中字符串的处理
1、Java字符串与原生字符串转换
当从java层传递一个字符串过来之后,它的类型是jstring,同样如果需要返回一个字符串给java层,它的类型也是jstring。jstring代表着Java虚拟机中的一个字符串,并且不同于C++语言的string类型。
如果原生代码需要处理jstring,需要通过JNIEnv将其转换为原生字符串才可以使用。通过JNI函数来读取这个字符串中的内容,函数可以通过JNIEnv接口指针调用,它将一个代表着Java虚拟机中的字符串jstring引用,转换成为一个UTF-8形式的C字符串。
当原生代码使用完了通过获取的原生字符串后应该使用释放它。调用标识着原生代码不再需要使用从获取的UTF-8字符串了,这个UTF-8字符串所占用的空间就可以被释放了。如果不调用释放原生字符串的话将会导致内存泄露。
我们看下函数的原型是:
const char* GetStringUTFChars(jstring string, jboolean* isCopy)
在这里第三个参数表示如果返回的字符串是原来的java.lang.String的一份拷贝,则在函数返回之后,isCopy指向的内存地址将会被设置为JNI_TRUE。而如果返回的字符串指针直接指向原来的java.lang.String对象,则该地址会被设置为JNI_FALSE.如果返回了JNI_FALSE, 则原生代码将不能改变返回的字符串,因为改变了这个字符串,原来的java字符串也会被修改,这违背了java.lang.String实例不可改变的原则。通常你可以直接传递NULL给isCopy来告诉Java虚拟机你不在乎返回的字符串是否指向原来Java的String对象。
如果需要将C/C++的字符串返回给Java层,则需要通过函数生成jstring返回。
例如下面的例子展示了在Native层获取java层字符串,并修改返回给java层的一个例子:
除了上面介绍的, 以及, JNI还提供了和等相关API处理Unicode格式的字符串。
下表是JNI中常用的一些操作字符串的相关API:
JNI中数组的处理
在JNI中使用jarray以及像jintArray等子类表示数组。正如jstring不是一个C/C++的字符串类型,jarray也不是C/C++的数组类型。如果需要在native处理数组,同样需要通过JNIEnv接口将jarray转换。
例如下面是一个展示了计算一个java数组之和的例子:
对于上述累计数组和的例子,使用函数获取数组元素实现也是可以的,但是需要注意的是要和配对使用,以免造成内存泄漏。
以下这个例子展示了在JNI函数中排序数组,然后将排序好的数组同步到java层的功能:
在上面的例子中和都可以获取到数组相关元素,那么他们有什么区别呢?
对于小量的、固定大小的数组,应该选择系列函数来操作数组元素是效率最高的。因为这对函数要求提前分配一个C/C++临时缓冲区来存储数组元素,开发者可以直接在栈上或在堆上来动态申请,当然在栈上申请是最快的。有童鞋可能会认为,访问数组元素还需要将原始数据全部拷贝一份到临时缓冲区才能访问而觉得效率低?其实这种复制少量数组元素的代价是很小的,几乎可以忽略。这对函数的另外一个优点就是,允许你传入一个开始索引和长度来实现对子数组元素的访问和操作(SetArrayRegion函数可以修改数组),不过传入的索引和长度不要越界,函数会进行检查,如果越界了会抛出 ArrayIndexOutOfBoundsException 异常。而系列函数永远是安全的,JVM会选择性的返回一个指针,这个指针可能指向原始数据,也可能指向原始数据的复制,更加适用于数据量比较大的数组。
下表是JNI中常用的一些操作数组的相关API:
JNI为访问对象数组单独提供了一组单独的函数返回指定下表的元素, 而则修改指定索引上的元素。对于引用类型数组与基础类型数组不同的是,你不能一次获取或者拷贝对象数组中的所有元素,需要使用Get/SetObjectArrayElement来访问引用类型的数组。
领取专属 10元无门槛券
私享最新 技术干货