Java和c/c++之间的结构体传输

JNI提供了Java和native代码相互调用的接口,注意是相互调用,不仅仅是Java可以调用native,native也是可以调用Java的。但是使用的时候,我们会遇到一些问题,本文介绍一下Java对象和底层结构体的转换。

Java 对象

我们有Person类,

public class Person {
    public int ID;
    public String name;
    public byte[] data;
}
 
 
  • 1
  • 2
  • 3
  • 4
  • 5

底层结构体

Student结构体,由于我们底层采用c实现,而c没有字符串类型(C++有),所以我们采用char数组来存储字符串。

typedef struct {
    int ID;
    char name[255];
    char data[255];

} Student;
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

我们来介绍,如何将Student对象传递到底层,并将其转换为底层Student结构体。

NDK配置

首先,我们新建一个JNIUtils类,用来从java调用底层代码,

public class JNIUtils {
    public static native void passJava2Native(Student persion);
    public static native Student getJavaFromNative();
    static {
        System.loadLibrary("java2struct");
    }
}
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

我们定义了两个本地方法,一个是将Java对象传递给底层,另一个是从底层返回一个Java对象。 
然后,我们执行javah -jni com.example.java2struct.JNIUtils命令来得到头文件,并得到两个本地方法的函数名,

JNIEXPORT void JNICALL Java_com_example_java2struct_JNIUtils_passJava2Native
  (JNIEnv *, jclass, jobject);
JNIEXPORT jobject JNICALL Java_com_example_java2struct_JNIUtils_getJavaFromNative
  (JNIEnv *, jclass);
 
 
  • 1
  • 2
  • 3
  • 4

底层接收Java对象

JNIEXPORT void JNICALL Java_com_example_java2struct_JNIUtils_passJava2Native(JNIEnv * env, jclass class, jobject object) 
 
 
  • 1

第三个参数就是我们传递进来的Student对象了。问题,就是,我们如何得到其各个成员变量的值呢? 
别急,JNI提供了GetFieldID方法,用来得到Java对象的FiledID,调用格式如下(C代码格式,C++代码有所不同),

(*env)->GetFieldID(env, clzz, fieldName, fieldSig);
 
 
  • 1

clzz为jclass类,可以通过(*env)->FindClass来获得,fieldName是成员的名称,fieldSig是成员的签名。关于类型签名,可以参考JNI java 类型签名 
得到ID后,我们根据成员的类型来决定调用以下方法,值得注意的是,Java的String也是类,所以可以通过GetObjectField来得到String类型的成员变量

GetObjectField, 
GetBooleanField, 
GetByteField, 
GetCharField, 
GetShortField, 
GetIntField, 
GetLongField, 
GetFloatField, 
GetDoubleField,

完整的代码如下,得到所有的成员变量后,我们将其值赋给结构体。

JNIEXPORT void JNICALL Java_com_example_java2struct_JNIUtils_passJava2Native(
        JNIEnv * env, jclass class, jobject object) {
    LOGE("call passJava2Native!");
    jclass jcRec = (*env)->FindClass(env, "com/example/java2struct/Student");
    jfieldID jfID = (*env)->GetFieldID(env, jcRec, "ID", "I");
    jfieldID jfname = (*env)->GetFieldID(env, jcRec, "name",
            "Ljava/lang/String;");
    jfieldID jfdata = (*env)->GetFieldID(env, jcRec, "data", "[B");
    int ID = (*env)->GetIntField(env, object, jfID);
    LOGE("ID: %d", ID);
    jstring name = (jstring)(*env)->GetObjectField(env, object, jfname);
    //convert jstring to char
    char* charname = (char*) (*env)->GetStringUTFChars(env, name, 0);
    LOGD("name: %s", charname);
    LOGE("name size : %d", strlen(charname));
    jbyteArray ja = (jbyteArray)(*env)->GetObjectField(env, object, jfdata);
    int nArrLen = (*env)->GetArrayLength(env, ja);
    char *chardata = (char*) (*env)->GetByteArrayElements(env, ja, 0);
    LOGW("data: %s", chardata);
    Student student;
    student.ID = ID;
    strcpy(student.name, charname);
    strcpy(student.data, chardata);
    printStudent(student);
}
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25

我们就可以得到相应的成员变量啦。然后我们就可以赋值给结构体。

底层如何返回Java对象

上面介绍了底层如何解析Java对象,这部分介绍底层如何返回一个Java对象,步骤如下, 
1. (*env)->FindClass得到类 
2. (*env)->AllocObject新建一个类的示例 
3. (*env)->GetFieldID得到域ID 
4. 设置域,方法如下,

SetObjectField, 
SetBooleanField, 
SetByteField, 
SetCharField, 
SetShortField, 
SetIntField, 
SetLongField, 
SetFloatField, 
SetDoubleField,

设置域(成员变量)后,然后将该对象返回即可。完整的代码如下,

JNIEXPORT jobject JNICALL Java_com_example_java2struct_JNIUtils_getJavaFromNative(
        JNIEnv * env, jclass class) {
    Student student;
    student.ID = 1234;
    char *charname = "Paul";
    char *chardata = "I am a student";
    strcpy(student.name, charname);
    strcpy(student.data, chardata);
    printStudent(student);

    LOGE("call getJavaFromNative!");
    jclass jcRec = (*env)->FindClass(env, "com/example/java2struct/Student");
    jfieldID jfID = (*env)->GetFieldID(env, jcRec, "ID", "I");
    jfieldID jfname = (*env)->GetFieldID(env, jcRec, "name",
            "Ljava/lang/String;");
    jfieldID jfdata = (*env)->GetFieldID(env, jcRec, "data", "[B");

    jobject joRec = (*env)->AllocObject(env, jcRec);

    (*env)->SetIntField(env, joRec, jfID, student.ID);

    //convert char to jstring
    jstring jstrn = (*env)->NewStringUTF(env, student.name);
    (*env)->SetObjectField(env, joRec, jfname, jstrn);

    //convert char to byte
    int length = strlen(student.data);
    jbyteArray jbarr = (*env)->NewByteArray(env, length);
    jbyte *jb = (*env)->GetByteArrayElements(env, jbarr, 0);
    memcpy(jb, student.data, length);
    (*env)->SetByteArrayRegion(env, jbarr, 0, length, jb);
    (*env)->SetObjectField(env, joRec, jfdata, jbarr);
    return joRec;
}
 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34

代码下载地址Github

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值