在写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对象,方法名称,方法签名。

方法签名由方法参数和方法返回值构成,这里又分为基本数据类型和引用类型,

基本数据类型对应关系表如下:

c交互 java so文件 java和c如何交互_string

引用类型规则如下:

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