在写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中构建对象,设置好属性后传过去。