在写JNI时java与C之间的数据传递是很寻常的,这里分析下集中java和C之间的传值情况。
1.java与C之间的基本数据传递。
在这之前有必要先了解java与C之间基本类型映射关系,如下表所示:
Java 类型 本地类型 描述
boolean jboolean C/C++8位整型
byte jbyte C/C++带符号的8位整型
char jchar C/C++无符号的16位整型
short jshort C/C++带符号的16位整型
int jint C/C++带符号的32位整型
long jlong C/C++带符号的64位整型e
float jfloat C/C++32位浮点型
double jdouble C/C++64位浮点型
Object jobject 任何Java对象,或者没有对应java类型的对象
Class jclass Class对象
String jstring 字符串对象
Object[] jobjectArray 任何对象的数组
boolean[] jbooleanArray 布尔型数组
byte[] jbyteArray 比特型数组
char[] jcharArray 字符型数组
short[] jshortArray 短整型数组
int[] jintArray 整型数组
long[] jlongArray 长整型数组
float[] jfloatArray 浮点型数组
double[] jdoubleArray 双浮点型数组
java和C之间的数据进行传递过程中都是需要进行转换的,转换关系就是上表。
举个从C返回字符串给Java的例子,C代码如下(这里是C++实现):
JNIEXPORT jstring JNICALL Java_perfecter_jni_LoadJni_printFromC(JNIEnv* env,
jobject obj) {
jstring str = env->NewStringUTF("Hello from JNI !");
return str;
}
对应的java本地方法声明如下:
public static native String printFromC();
这里就拿java的String与C的jstring举例了,上面 JNIEXPORT 和 JNICALL 是关键字。
2.java向C传递对象,C调用java对象的方法。
直接用代码说明,
JNIEXPORT jstring JNICALL Java_perfecter_jni_LoadJni_getStringFromJavaObj(
JNIEnv* env, jobject obj, jobject clsObj){
jmethodID methodId;
jclass objclass=env->GetObjectClass(clsObj);
methodId=env->GetMethodID(objclass,"getString","()Ljava/lang/String;");
jstring jstr=(jstring)env->CallObjectMethod(clsObj,methodId,NULL);
return jstr;
}
java native方法
public static native String getStringFromJavaObj(MyObj obj);
MyObj类:
package perfecter.jni;
public class MyObj
{
private String name;
public MyObj(){}
public String getString(){
return "String from MyObj"+name;
}
}
应该很快可以看出来,C中代码就是java中的反射机制。这里说明下,像GetObjectClass这些函数可以到sun的jni官方文档去查询。
说明下GetMethodID(objclass,"getString","()Ljava/lang/String;"),参数依次为方法对象的jclass对象,方法名称,方法签名。
方法签名由方法参数和方法返回值构成,这里又分为基本数据类型和引用类型,
基本数据类型对应关系表如下:
引用类型规则如下:
以“L”开头,以“;”结束,中间对应的是该类型的路径
如:String : Ljava/lang/String;
Object: Ljava/lang/Object;
自定义类 MyObj 对应 package perfecter.jni.MyObj;
MyObj : Lperfecter/jni/MyObj;
"."换成“/”。
数组表示: 数组表示的时候以“[” 为标志,一个“[”表示一个维度
如:int [ ] :[I
Long[ ][ ] : [[J
Object[ ][ ][ ] : [[[Ljava/lang/Object;
其实不知道怎么生成签名的可以直接用命令行去查看,具体做法是在生成的类的class目录下执行 javap –s MyObj.
3.C向java传递对象
直接看代码,
JNIEXPORT jobject JNICALL Java_perfecter_jni_LoadJni_getJavaObj(JNIEnv* env,
jobject obj){
jclass clazz=env->FindClass("perfecter/jni/MyObj");
if(clazz==0)
return 0;
jobject jobj=env->AllocObject(clazz);
jfieldID fieldId=env->GetFieldID(clazz,"name","Ljava/lang/String;");
env->SetObjectField(jobj,fieldId,env->NewStringUTF("wahaha"));
return jobj;
}
java对应的native方法申明public static native MyObj getJavaObj();
这个就是在C中构建对象,设置好属性后传过去。
最后附上JNI文档地址:http://java.sun.com/docs/books/jni/html/jniTOC.html