Android JNI 错误:NoSuchMethodError:没有非静态方法

김선달*_*김선달 5 c++ java java-native-interface android jnienv

我正在尝试做的简化如下。

  1. Java -> 调用 C++ 函数 A
  2. C++ 函数 A 调用 C++ 函数 B
  3. C++ 函数 B 调用 Java 方法 C

我必须存储 JVM(2) 和 global jobject(3)。

但在第 3 部分,

JNI DETECTED ERROR IN APPLICATION: JNI CallVoidMethodV called with pending exception java.lang.NoSuchMethodError: no non-static method "Lpackage/name/here/d/b;.setInput([F)V"
Run Code Online (Sandbox Code Playgroud)

我总是收到这个错误。


代码

结构

  • 包装器
  • 本机-lib.cpp
  • 包装器.cpp

** <-> 表示沟通

其他 Java 类 <-> Wrapper.java

Wrapper.java <-> native-lib.cpp

native-lib.cpp <-> Wrapper.cpp

Wrapper.cpp <-> 其他 C++ 类


包装器

private static long wrapperAddr = 0; // initializes later

private tfModel model;
private native void nativeSetModel(long native_ptr, tfModel model);

public Wrapper(...){
    ...

    model = tfModel.create(tfModel.Model.MNIST, tfModel.Device.CPU, 1);
    nativeSetModel(wrapperAddr, model);
}
Run Code Online (Sandbox Code Playgroud)

本机-lib.cpp

extern "C" JNIEXPORT void JNICALL
Java_package_name_here_jni_Wrapper_nativeSetModel(JNIEnv *env,
                                                      jobject instance,
                                                      jlong native_ptr,
                                                      jobject model){

  auto wrapper = reinterpret_cast<Wrapper *>(native_ptr);

  wrapper->setModel(env, model);
}
Run Code Online (Sandbox Code Playgroud)

包装器.cpp

void setModel(JNIEnv *env, jobject _model){
    env->GetJavaVM(&translater::jvm);    
    jobject gmodel = env->NewGlobalRef(_model);    
    translater::model = gmodel;
  }
Run Code Online (Sandbox Code Playgroud)

功能A

...
translater::setInputTS(input1, 100);  // error starts here

...
Run Code Online (Sandbox Code Playgroud)

translater.h / translater.cpp (函数 B )

class translater{
public:

    inline static JavaVM *jvm = nullptr;
    inline static jobject model = nullptr;

    // from: /sf/answers/2101836201/
    static bool GetJniEnv(JavaVM *vm, JNIEnv **env);  

    static void predictTS(std::vector<float> &output);

    static void translater::setInputTS(float* input, int len) {


    JNIEnv *env;
    bool did_attach = GetJniEnv(jvm, &env);

    if(did_attach){


        jclass clazz = env->GetObjectClass(model);
        jmethodID jid_input = env->GetMethodID(clazz, "setInput", "([F)V");  // Here is where error happens

    }

    ...    
};
Run Code Online (Sandbox Code Playgroud)

模型.java

public abstract class tfModel{
...

    public void setInput(float[] array){
        if(inputIndex < inputImageNum)
            setInput_Image(inputIndex, array);
        else
            setInput_Tensor(inputIndex - inputImageNum, array);

        ++inputIndex;
    }

}

Run Code Online (Sandbox Code Playgroud)

如何jmethodID jid_input = env->GetMethodID(clazz, "setInput", "([F)V"); 正确调用 ?

김선달*_*김선달 6

这是因为混淆器删除了未使用的代码,这有时可能是错误的。我添加了 proguard 设置,一切正常。

https://developer.android.com/studio/build/shrink-code#keep-code

。轮廓

...
// added
-keep abstract class package.name.here.module.tfModel{
    *;
}
Run Code Online (Sandbox Code Playgroud)