java代码:
public class Foo {
private long i = 0;
void printI() {
nativePrintI();
}
private native static void nativePrintI();
}
JNI 代码:
JNIEXPORT void JNICALL Java_com_aliyun_livestream_Publisher_nativeRelease
(JNIEnv *env, jobject obj) {
jclass cls = env->GetObjectClass(obj);;
jfieldID iField = env->GetFieldID(cls, "i", "J");
jlong i = env->GetLongField(obj, iField);
printf("%lld\n", i);
}
我的被调用者是这样的:
Foo foo = new Foo();
foo.printI();
有时它工作正常,但有时它会失败,并显示消息“”jni 对于 java.lang.Class 类的对象无效”
最佳答案
当您将native
方法声明为static
时,相应的JNI函数将接收一个JNIEnv*
和一个jclass
,加上您指定的任何其他参数(在本例中没有)。 jclass
参数是对调用该方法的类的引用。
这实际上与您在 Java 中进行的调用没有什么不同;当您调用静态
方法时,它将在类上调用,而不是在从该类实例化的对象上调用。
另一方面,非静态 native
方法将使用 JNIEnv*
和 jobject
调用 JNI 函数,其中 jobject
是对调用该方法的对象的引用。
在 C/C++ 中,jobject
和 jclass
都是指针,因此您可以将其中一个分配给另一个,但是当您尝试传递或者将它们返回到 Java/JNI 函数,这些指针指向哪种底层值很重要。
GetObjectClass
和 GetLongField
期望您向它们传递一个 jobject
,因此您在 中获得的
情况不起作用。jclass
static
如果您想要一个作业对象
,请将该方法声明为 Java 中的非静态方法。如果您想要一个 jclass
,请在 Java 中将该方法声明为静态。如果您想要两者,请将其声明为非静态,以便您收到一个 jobject
,然后您可以在其上调用 GetObjectClass
。
关于java - JNI GetLongField 失败,错误代码为 "jni not valid for an object of class java.lang.Class",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38260834/