13

我正在尝试使用 JNI 函数来创建 Java 类并使用 DeviceId.java 构造函数方法设置该类的一些属性。我可以使用 GetMethodID 获取构造函数方法,但是如何创建 Device.java 的新实例然后设置属性(setId 和 setCache)。目标是向调用者返回一个完全填充的 Device.java 对象实例。有任何想法吗?

JNI 功能:

 JNIEXPORT jobject JNICALL Java_com_test_getID(JNIEnv *env, jclass cls) 
    {
        jmethodID cnstrctr;
        jclass c = (*env)->FindClass(env, "com/test/DeviceId");
        if (c == 0) {
            printf("Find Class Failed.\n");
         }else{
            printf("Found class.\n");
         }

        cnstrctr = (*env)->GetMethodID(env, c, "<init>", "(Ljava/lang/String;[B)V");
        if (cnstrctr == 0) {
            printf("Find method Failed.\n");
        }else {
            printf("Found method.\n");
        }

        return (*env)->NewObject(env, c, cnstrctr);
    }

Java 类:

package com.test;

public class DeviceId {
    private String id;
    private byte[] cache;

        public DeviceId(){}
    public DeviceId(String id, byte[] cache){
        this.id=id;
        this.cache=cache;
    }

    public byte[] getCache() {
        return cache;
    }

    public void setCache(byte[] cache) {
        this.cache = cache;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }   
}
4

2 回答 2

10

当您调用 时GetMethodID,您为双参数构造函数提供了签名。因此,您只需要在调用时传递您的jstring和 a - 例如:jbytearrayNewObject

return (*env)->NewObject(env, c, cnstrctr, id, cache);

除非您决定调用 0-arg 构造函数,否则您不需要调用setIdandsetCache方法——这只会使您的代码复杂化,因为您必须GetMethodID调用它们并调用它们。继续沿着你正在走的路线更简单。

于 2012-04-12T20:05:54.407 回答
0

我想将JNI 的 cpp 代码中的自定义Java 对象返回给 Java。解决方案是从 cpp 函数返回一个并在本地方法声明中使用我们自定义的 Java 对象。jobject

public class PyError {

    public String message;

    public boolean occurred;

    public PyError(boolean occurred, String message){
        this.message = message;
        this.occurred = occurred;
    }
} 

和Java中的方法声明:

native PyError nativePythonErrorOccurred();

在 cpp 方面:

extern "C" JNIEXPORT jobject JNICALL
Java_com_your_package_nativePythonErrorOccurred(JNIEnv *env, jobject obj) {

    jclass javaLocalClass = env->FindClass("com/your/package/PyError");

    if (javaLocalClass == NULL) {
        LOGP("Find Class Failed.\n");
    } else {
        LOGP("Found class.\n");
    }

    jclass javaGlobalClass = reinterpret_cast<jclass>(env->NewGlobalRef(javaLocalClass));

    // info: last argument is Java method signature
    jmethodID javaConstructor = env->GetMethodID(javaGlobalClass, "<init>", "(ZLjava/lang/String;)V");

    if (javaConstructor == NULL) {
        LOGP("Find method Failed.\n");
    } else {
        LOGP("Found method.\n");
    }

    jobject pyErrorObject = env->NewObject(javaGlobalClass, javaConstructor, true, env->NewStringUTF("Sample error body"));
    return pyErrorObject;
}

使用 确定方法的签名 javap -s java.your.package.YourClass。另外,看看这里

如果您遇到类似以下的错误:JNI ERROR (app bug): attempt to use stale Global 0xf2ac01ba您的方法签名错误,您传递了错误的参数,env->NewObject()或者您没有使用 jni 对象的全局状态 - 更多信息请点击此处

于 2017-04-12T08:20:28.867 回答