JNI java和c之间对象的传递

在写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



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章