转自: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