java - JNI GetLongField 失败,错误代码为 "jni not valid for an object of class java.lang.Class"

标签 java android android-ndk java-native-interface

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++ 中,jobjectjclass 都是指针,因此您可以将其中一个分配给另一个,但是当您尝试传递或者将它们返回到 Java/JNI 函数,这些指针指向哪种底层值很重要。
GetObjectClassGetLongField 期望您向它们传递一个 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/

相关文章:

java - Quicksort:wikipedia-implementation 不工作

java - 计算你借了多少次

java - 如何在android中将线性布局的所有元素设置为相同的宽度?

android - 从 Android JNI 程序调用的 Log API 是什么?

安卓.mk : Unsupported source file extensions for C files

java - 传递对方法的引用 Java

java - Dropwizard JWT 工作流程

android - 如何在 Android 应用程序中显示动画 GIF 图像?

android - 如何取消应用程序与谷歌驱动器帐户的链接

android-ndk - Android,CMake和静态链接