有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java Android JNI在应用程序中检测到错误:调用JNI GetMethodID时出现挂起异常

我正试图用我的安卓项目运行谷歌OCR Tesseract。我已经将tesseract与安卓 ndk兼容,并在尝试运行安卓项目后收到此错误

我的环境如下

  • 安卓5.1.1
  • 适用于windows的安卓-ndk-r10e
  • 安卓-sdk-r22

作为参考,我根据这里列出的一个示例构建Example Link

提前谢谢

以下是我的logcat结果的一个片段:

  I/DEBUG   (  182): Revision: '0'
    I/DEBUG   (  182): ABI: 'arm'
    I/DEBUG   (  182): pid: 20291, tid: 20337, name: JavaBridge  >>> com.enterprisem
    obility.OCR <<<
    I/DEBUG   (  182): signal 6 (SIGABRT), code -6 (SI_TKILL), fault addr --------
    I/DEBUG   (  182): Abort message: 'art/runtime/check_jni.cc:65] JNI DETECTED ERR
    OR IN APPLICATION: JNI GetMethodID called with pending exception 'java.lang.NoSu
    chFieldError' thrown in void com.googlecode.tesseract.安卓.TessBaseAPI.native
    ClassInit():-2'
    I/DEBUG   (  182):     r0 00000000  r1 00004f71  r2 00000006  r3 00000000
    I/DEBUG   (  182):     r4 a0701db8  r5 00000006  r6 0000000b  r7 0000010c
    I/DEBUG   (  182):     r8 00000000  r9 b486f520  sl a1c0ac00  fp 00000001
    I/DEBUG   (  182):     ip 00004f71  sp a07006d8  lr b6e503c5  pc b6e72f6c  cpsr
    60070010
    I/DEBUG   (  182):
    I/DEBUG   (  182): backtrace:
    I/DEBUG   (  182):     #00 pc 00039f6c  /system/lib/libc.so (tgkill+12)
    I/DEBUG   (  182):     #01 pc 000173c1  /system/lib/libc.so (pthread_kill+52)
    I/DEBUG   (  182):     #02 pc 00017fd3  /system/lib/libc.so (raise+10)
    I/DEBUG   (  182):     #03 pc 00014795  /system/lib/libc.so (__libc_安卓_abor
    t+36)
    I/DEBUG   (  182):     #04 pc 00012f44  /system/lib/libc.so (abort+4)
    I/DEBUG   (  182):     #05 pc 00228cd7  /system/lib/libart.so (art::Runtime::Abo
    rt()+170)
    I/DEBUG   (  182):     #06 pc 000a7371  /system/lib/libart.so (art::LogMessage::
    ~LogMessage()+1360)
    I/DEBUG   (  182):     #07 pc 000b1b17  /system/lib/libart.so (art::JniAbort(cha
    r const*, char const*)+1118)
    I/DEBUG   (  182):     #08 pc 000b2055  /system/lib/libart.so (art::JniAbortF(ch
    ar const*, char const*, ...)+68)
    I/DEBUG   (  182):     #09 pc 000b530f  /system/lib/libart.so (art::ScopedCheck:
    :ScopedCheck(_JNIEnv*, int, char const*)+1346)
    I/DEBUG   (  182):     #10 pc 000b7755  /system/lib/libart.so (art::CheckJNI::Ge
    tMethodID(_JNIEnv*, _jclass*, char const*, char const*)+36)
    I/DEBUG   (  182):     #11 pc 001332f7  /data/app/com.enterprisemobility.OCR-1/l
    ib/arm/libtess.so (Java_com_googlecode_tesseract_安卓_TessBaseAPI_nativeClass
    Init+46)
    I/DEBUG   (  182):     #12 pc 0000614d  /data/dalvik-cache/arm/data@app@com.ente
    rprisemobility.OCR-1@base.apk@classes.dex
    W/ActivityManager(  536):   Force finishing activity 1 com.enterprisemobility.OC
    R/.MainActivity
    I/DEBUG   (  182):
    I/DEBUG   (  182): Tombstone written to: /data/tombstones/tombstone_07

共 (5) 个答案

  1. # 1 楼答案

    您应该使用android ndk中的ndk堆栈工具来找出崩溃的位置。请参阅关于ndk stack的链接

    adb logcat > /tmp/foo.txt
    
    $NDK/ndk-stack -sym $PROJECT_PATH/obj/local/armeabi -dump foo.txt
    
  2. # 2 楼答案

    调用GetMethodID时遇到类似问题:

    JNI DETECTED ERROR IN APPLICATION: thread Thread[1,tid=15092,Runnable,Thread*=0x791c1b8000,peer=0x7278ab58,"main"] using JNI after critical get
    ...   in call to GetMethodID
    

    使用GetMethodID之前的几个调用

    inBufP = env->GetPrimitiveArrayCritical(jIn, NULL);
    

    解决方案: 我可以通过将GetPrimitiveArray更改为GetByteArrayElements

    jboolean isCopy;
    inBufP = env->GetByteArrayElements(jIn, &isCopy);
    
  3. # 3 楼答案

    我也有同样的问题,这让我困惑了两天。最后一个原因是我传递了错误的对象类型。例如 java代码是

    public OverlayLine(int mWidth,List<GeoPoint> mPoints);
    

    我注册jni方法如下:

    gClass.mInitMethod = env->GetMethodID(gObject, "<init>", "(ILjava/lang/Object;)V");
    

    并在遇到错误时获取错误消息。我修正了密码

    gClass.mInitMethod = env->GetMethodID(gObject, "<init>", "(ILjava/util/List;)V");
    

    错误消失了。您应该传递精确的对象类型,而不是“Ljava/lang/object;”

  4. # 4 楼答案

    <>最可能的是,JNI映射在C++代码中被错误地定义。JNI对Java的类型映射有非常严格的约定。例如 在从JNI调用Java对象的方法之前,我们需要它的签名。因此,该方法:

    long myMethod (int n, String s, int[] arr);

    从JNI中可以看到带有签名的:

    (ILJAVA/LANG/STRING;[I])J

    您可以在此处阅读这些规则的全面概述: http://www.rgagnon.com/javadetails/java-0286.html

  5. # 5 楼答案

    中止消息相对清晰:您调用GetFieldID(cls, fieldName)获取传递给此函数的类中不存在的字段名,但不检查该错误,并继续调用其他JNI函数。不幸的是,您不能忽略这些错误。在调用GetMethodID()或大部分JNI函数之前,必须调用ExceptionClear()

    您可以使用addr2line查找对getMethodID()的哪个特定调用崩溃,并基于此,派生出对GetFieldID(cls, fieldName)的哪个调用失败。但我建议将错误检查添加到所有JNI调用中,因为明天其他函数可能会抛出异常