JNI中多维数组的处理

时间:2022-04-21 19:00:25

转自:http://developers.sun.com.cn/blog/joeys/entry/200703091

 

最近有朋友问到我,在JNI中,如何来处理多维Java数组,还给了我一个处理String多维数组的实际程序。

那么,问题的关键在于这些多维数组在JNI中是以什么样的形式传递的呢。来看一个例子:

native static void nativePrintArray(String[] array);

native static void nativePrintArray(String[][] arrray);

这两个Java的Native函数,在javah编译出来的头文件中有什么不同呢?

我们注意到,这两个函数的JavaSignature是不同的,一个是[Ljava/lang/String;)V,另外一个是[[Ljava/lang/String;)V,但是在native的.h文件中,两个Native函数的入口参数类型是完全一样的,数组对应的参数都是jobjectArray。当然,我们知道对于二维数组,这个jobjectArray中的元素本身也是jobjectArray。

在JNI的接口函数中,如下函数用以得到数组中的某个元素

jobject (JNICALL *GetObjectArrayElement) (JNIEnv *env, jobjectArray array, jsize index)

那么在多维数组的情况下,问题是,我们能否将得到的jobject转化成jobjectArray呢?参看jni.h头文件,对于C语言的情况,我们看到如下的定义:

typedef jobject jarray;

typedef jarray jobjectArray;

因此,我们知道,我们可以无缝的将我们从多维数组得到的jobject对象转换成jobjectArray。


以下是一个在native代码中打印二维String数组的元素的源代码,更高维的情况,只是增加了递归的层次而已:

Java Code:

public class TestArray {
static {
System.loadLibrary("test");
}

public static native void PrintArrays(String[][] Info);

public static void main(String[] args) {
String[][] info = {
{"INFO_00", "INFO_01", "INFO_02"},
{"INFO_10", "INFO_11"}
};

PrintArrays(info);
}
}

Native Code:

#include "TestArray.h"

JNIEXPORT void JNICALL Java_TestArray_PrintArrays
(JNIEnv *env, jclass class, jobjectArray array)
{
int size = (*env)->GetArrayLength(env, array);
for (int i = 0; i < size; i++) {
jarray myarray = ((*env)->GetObjectArrayElement(env, array, i));
int leng = (*env)->GetArrayLength(env, myarray);
printf("length: %dn", leng);
for (int j = 0; j < leng; j++) {
jstring string = ((*env)->GetObjectArrayElement(env, myarray, i));
const char * chars = (*env)->GetStringUTFChars(env, string, 0);
printf("%s n", chars);
(*env)->ReleaseStringUTFChars(env, string, chars);
}
}
}

在Solaris下,编译运行命令如下:

#javac TestArray.java

#javah TestArray

#cc -G -I/usr/java/include -I/usr/java/include/solaris -o libtest.so TestArray.c

#export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH

#java TestArray