我发现这段代码非常困难(尤其是因为我一周前才开始玩C)。
我一直在努力寻找正确的语法来正确创建 C 中的 java 字符串数组(即 jstring 对象的数组,即表示 jstring 对象数组的对象)。我一直在使用以下资源,并从中构建了可编译的代码。我不确定之后发生的错误是由于语法不正确还是由于完全独立的原因。由于代码大多是孤立的,我假设语法不正确。
(Suns Native Programming Documentation & Suns JNI 文档)
代码可以编译,但在传递"FindClass"代码行后,发送一个SIGSEGV信号,该信号会杀死C进程:
jint size = 5;
jclass StringObject = (*env)->FindClass(env, "java/lang/String");
jobjectArray sampleMessage = (*env)->NewObjectArray(env, size, StringObject, NULL);
jobjectArray returnArray = (jobjectArray) (*env)->NewObjectArray(env, messageCount, &sampleMessage, 0);
谁能指出我一个有用的资源?或确认语法正确。
编辑
我的大部分问题是调试此代码导致了问题。我没有时间缩小重现因素的范围,但是通过 eclipse 在 gdb 客户端中跳过 JNI 代码不起作用。
要获取行类型的 jclass,可以在其中一行上调用 GetObjectClass()
。 这有效:
主.java
public class Main {
static {
System.loadLibrary("mynative");
}
private static native String[][] getStringArrays();
public static void main(String[] args) {
for (String[] array : getStringArrays())
for (String s : array)
System.out.println(s);
}
}
Mynative.c
static jobjectArray make_row(JNIEnv *env, jsize count, const char* elements[])
{
jclass stringClass = (*env)->FindClass(env, "java/lang/String");
jobjectArray row = (*env)->NewObjectArray(env, count, stringClass, 0);
jsize i;
for (i = 0; i < count; ++i) {
(*env)->SetObjectArrayElement(env, row, i, (*env)->NewStringUTF(env, elements[i]));
}
return row;
}
JNIEXPORT jobjectArray JNICALL Java_Main_getStringArrays(JNIEnv *env, jclass klass)
{
const jsize NumColumns = 4;
const jsize NumRows = 2;
const char* beatles[] = { "John", "Paul", "George", "Ringo" };
jobjectArray jbeatles = make_row(env, NumColumns, beatles);
const char* turtles[] = { "Leonardo", "Raphael", "Michaelangelo", "Donatello" };
jobjectArray jturtles = make_row(env, NumColumns, turtles);
jobjectArray rows = (*env)->NewObjectArray(env, NumRows, (*env)->GetObjectClass(env, jbeatles), 0);
(*env)->SetObjectArrayElement(env, rows, 0, jbeatles);
(*env)->SetObjectArrayElement(env, rows, 1, jturtles);
return rows;
}
为清楚起见,省略了构建、错误处理。