在我的情况下,使用 JNI 调用 CallVoidMethod 时抛出 NPE 的可能原因是什么

What's the possible reason why NPE is throwed when call CallVoidMethod with JNI in my situation

本文关键字:NPE 是什么 情况下 我的 使用 JNI CallVoidMethod 调用      更新时间:2023-10-16

场景:我已经用Cpp构建了一个DLL,它被带有JNI的java应用程序使用。DLL也将调用Java中的方法。然而,当我调用CallVoidMethod时,java控制台中将打印一个java.lang.NullPointerException

代码如下:

void CaptureCallback::ACallbackMethod(){
    jvm->AttachCurrentThread((void **)&env, NULL);
    jmethodID mid= env->GetMethodID(cls, "helloWorld", "()V");
    cout << "The jmethodID is : " << mid << endl;
    cout << "The jobject is : " << jobj << endl;
    cout << "The env is: " << env << endl;
    env->CallVoidMethod(obj, mid );
    jvm->DetachCurrentThread();
}

到目前为止我尝试过的:

  1. jmethod的值是而不是零。这意味着我已经找到了Java端的方法。(如果我错了,请纠正我。)
  2. 两个jclass-cls=env->GetObjectClass(jobj);和jclass-cls=env->FindClass(absolute_path)试图获取jclass信息

堆栈跟踪非常简单,它是:

线程"线程-2"出现异常

java.lang.NullPointerException

所以我很难找到原因。

当前进度:当我将ACallbackMethod的主体放入Constructor方法时,我成功地调用了helloWorld方法。构造函数方法如下所示:

CaptureCallback::CaptureCallback(JNIEnv *p_env, jobject p_jobj){
        env = p_env;
        jobj = p_jobj;
        jvm->AttachCurrentThread((void **)&env, NULL);
        jmethodID mid= env->GetMethodID(cls, "helloWorld", "()V");
        cout << "The jmethodID is : " << mid << endl;
        cout << "The jobject is : " << obj << endl;
        cout << "The env is: " << env << endl;
        env->CallVoidMethod(obj, mid );   
        jvm->DetachCurrentThread();
}

CaptureCallback是一个继承自IDL生成的头文件的类,CaptureCallback::ACallbackMethod保持从本机端调用。未声明的变量(如env、jobj等)在CaptureCallback.h文件中声明。

与上次提问的答案相同。obj要么为null,要么在被调用的方法或其调用的方法中,某个东西在为null时被取消引用,依此类推。或者,有人,可能是你给出了一些现在已经删除的代码,出于某种可能无关的原因故意抛出这个异常。

【请注意:这些确实是唯一可能的原因。如果你不同意,请给出你选择的理由。】

遗憾的是,您的代码没有错误检查。您需要检查每个JNI调用的结果,而不仅仅是您喜欢的调用。之前的任何一个都可能抛出NPE。我在上面依赖于你的断言,即它是CallVoidMethod(),但它可能不是。

为什么你要把一个可能是任何东西的异常变成NullPointerException是个谜。您应该重新抛出throwable,而不是抛出一个新的、可能不相关的异常来混淆自己。