繁体   English   中英

如何在Android NDK上使用JNI在C和Java之间传递复杂的结构

[英]How to pass a complex structure between C and Java with JNI on Android NDK

我的Android应用程序中的C代码有一个复杂的结构,我想在Java端使用它。 我已经在Google和stackoverflow上进行了一些研究,因此我已经从C结构创建了Java类,但是现在如何在Java中获取它。

我已经找到了有关在类中创建指针并在C端使用它的信息:

Get the field ID : (*env)->GetFieldID(...)
Get the pointer : (*env)->GetLongField(...)
Set the pointer : (*env)->SetLongField(...)

但是我不明白它是如何工作的...

在上面,您可以找到我到目前为止所做的……不算多! 在C端:

ComplexStructure Java_com_main_MainActivity_listenUDP(JNIEnv* env, jclass clazz)
{
    int i,taille;
    ComplexStructure myStruct;    
    taille = -1;    
    taille = recvfrom(socket, &myStruct, sizeof(ComplexStructure ), 0, &rcvAddr, &sizeOfSock);
    if(taille != -1)
    {   
        return myStruct;
    }
    return NULL;
}

在Java方面:

public void getFromUDP() {

    ComplexClass myClass = new ComplexClass();
    myClass = listenUDP();              
}

@Override
public void run() {
    initUDP();
    getFromUDP();
}


public static native ComplexClass listenUDP();
public static native void initUDP();
public static native void closeUDP();

/** Load jni .so on initialization */
static {
     System.loadLibrary("native-interface");
}

编辑:我想补充一点,我的结构像这样非常复杂:

typedef struct{
  TYPE_A myStructA;
  TYPE_B myStructB;
  TYPE_C myStructC;
  TYPE_D myStructD;
}ComplexStructure;

typedef struct{
  float rad;
  int size;
  bool isEmpty;
}TYPE_A;

typedef struct{
  float rad;
  bool isEmpty;
  float color;
  int temp;
}TYPE_B;

typedef struct{
  int temp;
  float rain;
  bool isEmpty;
}TYPE_C;

typedef struct{
  float rad;
  int idPerson;
  bool isOnTime;
}TYPE_D;

更复杂的是,仅是一个示例向您展示它的状态!

您不能将原始C结构传递给Java,并期望它将这些结构视为类。 您需要为您的结构创建一个类。 我看到您已经做到了,所以您唯一需要做的就是将这个结构转换为该类的实例。

Java方面的代码:

public static native ComplexClass listenUDP();

将转换为:

JNIEXPORT jobject JNICALL Java_com_main_MainActivity_listenUDP(JNIEnv *env, jclass);

在该C代码中,您需要使用env->FindClass();来加载ComplexClass env->FindClass(); 功能。 然后,要创建该类的新实例(如果您使用零参数的构造函数,它将简化操作),您需要加载一个构造函数方法签名,并在env->NewObject()方法中“调用”它。 完整代码:

jclass complexClass = env->FindClass("/com/main/ComplexClass");
jmethod constructor = env->GetMethodId(complexClass, "<init>", "()com/main/ComplexClass"); //The name of constructor method is "<init>"
jobject instance = env->NewObject(complexClass, constructor);

然后,您需要使用env->setXXXField();来设置此类的字段env->setXXXField(); 如果您有更多对象作为字段并且还想创建它们,则对另一个对象重复上述过程。

这看起来很复杂,但这就是在托管Java代码中使用本机C的代价。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM