Android apk JNI отчет не может найти метод

Ну, мое приложение Java-код выглядит следующим образом:

package doc.android.demo;

public class NativeInterface {


    private String mStrCrtMsg;
    /** The current file name to be displayed */
    private String mStrCrtFileName;
    public static final native int NtvPerformADT(int test, String path);
    public static final native int registerImage(char[] szDarkPlane0, char[] szDarkPlane1, 
            char[] szLightPlane0, char[] szLightPlane1);
    public static final native int getOutPut(char[] szHdrImg);

    /** this method is called from the JNI code */
    public void DisplayMessage(String strMessage) {
        mStrCrtMsg = mStrCrtMsg + "\n" + strMessage;
        System.out.println("huangzhiquan app DisplayMessage" + mStrCrtMsg);

    }

    /** this method is called from the JNI code */
    public void DisplayFileName(String strFileName) {   
        mStrCrtFileName = strFileName;
        System.out.println("huangzhiquan app DisplayFileName" + mStrCrtFileName);

    }
}

и мой код JNI:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
#include "ADT.h"
/* Header for class doc_android_demo_NativeInterface */

#ifndef _Included_doc_android_demo_NativeInterface
#define _Included_doc_android_demo_NativeInterface


#define TRUE 1
#define FALSE 0

#define NULL 0

// Logging support
#if defined(ANDROID) && defined(LOGGING)
#include <android/log.h>
#else
#include <android/log.h>
//#define __android_log_print(...) do{}while(0)
#endif

JNIEnv * theEnv = 0;
jobject * theObj = 0;
jmethodID midDM;
jmethodID midDFN;


/*
 * Class:     doc_android_demo_NativeInterface
 * Method:    NtvPerformADT
 * Signature: (ILjava/lang/String;)I
 */
JNIEXPORT jint JNICALL Java_doc_android_demo_NativeInterface_NtvPerformADT
  (JNIEnv * env, jobject thiz, jint test, jstring strLibsDirectory){
     theEnv = env;
        theObj = & thiz;

        jclass cls;

        cls = ( * env)->GetObjectClass (env, thiz);
        __android_log_print(ANDROID_LOG_ERROR, "huangzhiquan class name", cls);

        midDM = ( * env)->GetMethodID (env, cls, "DisplayMessage", "(Ljava/lang/String;)V");
        {
            __android_log_print (ANDROID_LOG_ERROR, "doc/android/demo", " Can't find method DisplayMessage");
            return 1;
        }

        midDFN = ( * env)->GetMethodID (env, cls, "DisplayFileName", "(Ljava/lang/String;)V");

        if (midDFN == 0)
        {
            __android_log_print (ANDROID_LOG_ERROR, "doc/android/demo", "Can't find method DisplayFileName");
            return 1;
        }

        jboolean isCopy;
        const char * pszLibsDir = ( * env)->GetStringUTFChars ( env, strLibsDirectory, & isCopy);
        int retval = PerformADT (test, pszLibsDir);
        // // Free up memory to prevent memory leaks
        ( * env)->ReleaseStringUTFChars (env, strLibsDirectory, pszLibsDir);

        theEnv = 0;
        theObj = 0;
        midDM  = 0;
        midDFN = 0;

        return (jint) retval;
}

/*
 * Class:     doc_android_demo_NativeInterface
 * Method:    registerImage
 * Signature: ([C[C[C[C)I
 */
JNIEXPORT jint JNICALL Java_doc_android_demo_NativeInterface_registerImage
  (JNIEnv * env, jobject thiz, jcharArray darkplane0, jcharArray darkPlane1, jcharArray lightPlane0, jcharArray lightPlane1){
    int retval = 0;
    return (jint) retval;
}

/*
 * Class:     doc_android_demo_NativeInterface
 * Method:    getOutPut
 * Signature: ([C)I
 */
JNIEXPORT jint JNICALL Java_doc_android_demo_NativeInterface_getOutPut
  (JNIEnv * env, jobject thiz, jcharArray hdrout){
    jcharArray charArr;
    return charArr;
}


void Java_DisplayMessage (const char * pszMsg)
{
    jstring strMsg;
    jclass cls;
    JNIEnv * env = theEnv;
    jobject jobj = * theObj;

    strMsg = ( * env)->NewStringUTF (env, pszMsg);

    if ( strMsg != NULL)
    {
        ( * env)->ExceptionClear (env);
        ( * env)->CallVoidMethod (env, jobj, midDM, strMsg);

        if ( ( * env)->ExceptionOccurred (env) )
        {
            __android_log_print (ANDROID_LOG_ERROR, "doc/android/demo", "error occured calling DisplayMessage");
            ( * env)->ExceptionDescribe (env);
            ( * env)->ExceptionClear (env);
        }

        ( * env)->DeleteLocalRef (env, strMsg);
    }
    else
        __android_log_print (ANDROID_LOG_ERROR, "doc/android/demo", "string could not be allocated");
}
//_____________________________________________________________

void Java_DisplayFileName (const char * pszMsg)
{
    jstring strMsg;
    jclass cls;
    JNIEnv * env = theEnv;
    jobject jobj = * theObj;

    strMsg = ( * env)->NewStringUTF (env, pszMsg);

    if ( strMsg != NULL)
    {
        ( * env)->ExceptionClear (env);
        ( * env)->CallVoidMethod (env, jobj, midDFN, strMsg);

        if ( ( * env)->ExceptionOccurred (env) )
        {
            __android_log_print (ANDROID_LOG_ERROR, "doc/android/demo", "error occured calling DisplayFileName");
            ( * env)->ExceptionDescribe (env);
            ( * env)->ExceptionClear (env);
        }

        ( * env)->DeleteLocalRef (env, strMsg);
    }
    else
        __android_log_print (ANDROID_LOG_ERROR, "doc/android/demo", "string could not be allocated");
}

#endif

Я не знаю, почему он всегда сообщает, хотя я старался изо всех сил. Кто-нибудь может мне помочь?

09-17 18:05:19.848 E/AndroidRuntime( 7736): FATAL EXCEPTION: AndroidDemo thread

09-17 18:05:19.848 E/AndroidRuntime( 7736): java.lang.NoSuchMethodError: no method with name='DisplayMessage' signature='(Ljava/lang/String;)V' in class Ljava/lang/Class;

09-17 18:05:19.848 E/AndroidRuntime( 7736):     at doc.android.demo.NativeInterface.NtvPerformADT(Native Method)

09-17 18:05:19.848 E/AndroidRuntime( 7736):     at doc.android.demo.ADTActivity.PerformADT(ADTActivity.java:81)

09-17 18:05:19.848 E/AndroidRuntime( 7736):     at doc.android.demo.ProcessingThread.run(ProcessingThread.java:36)

2 ответа

Решение
cls = ( * env)->GetObjectClass (env, thiz);

возвращает класс Class, как указано NoSuchMethodError (...) in class Ljava/lang/Class;

Это потому, что ваш родной метод является статическим, поэтому нет jobject быть переданным нативному методу, а скорее jclass, Ваша подпись должна быть:

JNIEXPORT jint JNICALL Java_doc_android_demo_NativeInterface_NtvPerformADT
   (JNIEnv * env, jclass cls, jint test, jstring strLibsDirectory){

Затем вы можете использовать cls напрямую.

Тем не менее, вы должны заметить, что, находясь в статическом методе, нет объекта для вызова метода, который вы найдете таким образом. (DisplayMessage будучи не статичным)

Я не уверен, что вы пытаетесь сделать, и я не вижу звонка NtvPerformADT который, вероятно, генерирует ошибку, но может ли это иметь какое-то отношение к функции с именем Java_DisplayMessage в вашем коде JNI? Разве это не должно быть DisplayFileName вместо?

Другие вопросы по тегам