调用JNI函数创建对象

时间:2019-05-18 18:46:23

标签: java c jvm language-lawyer

我正在用C11编写JNI人员,并且对严格遵循堆上对象创建有疑问。

JNI API提供了具有以下签名的功能:

jobject NewObject(JNIEnv *env, jclass clazz, jmethodID methodID, ...);

根据6.5.2.2(p7)标准

中的规定
  

函数原型声明符中的省略号引起   参数类型转换将在最后声明的参数之后停止。

对应于省略号的参数应显式转换为期望的类型,以使代码符合要求。考虑以下情况:

public class Event{
    public final int eventType;
    public final String meta;

    public Event(int eventType, String meta){
        this.eventType = eventType;
        this.meta = meta;
    }
}

我应该将与省略号对应的参数转换为哪种类型的参数?

我猜应该看起来像这样:

jclass event_class = ((*env)->FindClass)(env, "f/q/c/n/Event");
jmethodID ctor = (*env)->GetMethodID(
    env, 
    event_class, 
    "<init>", 
    "(ILjava/lang/String;)V"
);
array_element = (*env)->NewObject(
    env, 
    event_class, 
    ctor, 
    (jint) 0, (jobject) NULL //corresponds to the ellipsis
);

1 个答案:

答案 0 :(得分:1)

自调用的方法推导出参数的类型。

在您的情况下,Event类的构造函数需要一个int和一个String。

所以它看起来像这样:

jstring metaStr = (*env)->NewStringUTF(env, "hello");

jobject array_element = (*env)->NewObject(
        env,
        event_class,
        ctor,
        (jint)4711, metaStr
);

测试

要执行简短的测试,我们可以编写一个类,该类调用一个本机C函数,该函数创建所需的Event对象,对其进行初始化,然后将其返回给调用Java的一侧。

此Java程序如下所示:

import f.q.c.n.Event;

public class JNIEventTest {

    static {
        System.loadLibrary("native");
    }

    private native Event retrieveNativeEvent();


    public static void main(String[] args) {
        JNIEventTest jniEventTest = new JNIEventTest();
        Event event = jniEventTest.retrieveNativeEvent();
        System.out.println("eventType: " + event.eventType);
        System.out.println("meta: " + event.meta);
    }
}

然后,本机C端将如下所示:

#include "JNIEventTest.h"

JNIEXPORT jobject JNICALL Java_JNIEventTest_retrieveNativeEvent(JNIEnv *env, jobject thisObject) {


    jclass event_class = ((*env)->FindClass)(env, "f/q/c/n/Event");
    jmethodID ctor = (*env)->GetMethodID(
            env,
            event_class,
            "<init>",
            "(ILjava/lang/String;)V"
    );

    jstring eventStr = (*env)->NewStringUTF(env, "hello");

    jobject array_element = (*env)->NewObject(
            env,
            event_class,
            ctor,
            (jint)4711, eventStr
    );

    return array_element;
}

控制台中的调试输出如下:

eventType: 4711
meta: hello