我正在用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
);
答案 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