如何在Android NDK上使用JNI在C和Java之间传递复杂的结构
问题内容:
我的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->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();
。如果您有更多对象作为字段并且还想创建它们,则对另一个对象重复上述过程。
这看起来很复杂,但这就是在托管Java代码中使用本机C的代价。