如何从java本机接口调用getStackTrace方法(jni)

时间:2018-04-23 21:00:29

标签: java java-native-interface

我试图用一些例子来理解jni。我正在尝试使用jni获取java堆栈跟踪,所以这就是我正在做的事情

HelloJNI.java

package test.com.jni;

public class HelloJNI {
    static {
        System.loadLibrary("hello"); // Load native library at runtime
    }

    private native StackTraceElement[] getStackTraceNative(Throwable throwable);
    private native void printStackTraceNative(Throwable throwable);

    public static void main(String[] args) {
        test();
    }

    public static void test() {
        new HelloJNI().printStackTraceNative(new Throwable()); //Invoke native method
        new HelloJNI().getStackTraceNative(new Throwable());
    }
}

本机代码(为简单起见,保持错误处理)

test_com_jni_HelloJNI.c

JNIEXPORT jobjectArray JNICALL Java_test_com_jni_HelloJNI_getStackTraceNative (JNIEnv * env, jobject object, jthrowable exception) {

    jclass exceptionClazz = (*env)->GetObjectClass(env, exception); 

    jmethodID getStackTraceMethod = (*env)->GetMethodID(env, exceptionClazz, "getStackTrace", "()[Ljava.lang.StackTraceElement");

    jobjectArray stacktraces = (*env)->CallObjectMethod(env, exception, getStackTraceMethod);

    return stacktraces;
}

JNIEXPORT void JNICALL Java_test_com_jni_HelloJNI_printStackTraceNative (JNIEnv * env, jobject object, jthrowable exception) {

    jclass exceptionClazz = (*env)->GetObjectClass(env, exception); // can't fail

    jmethodID printStackTraceMethod = (*env)->GetMethodID(env, exceptionClazz, "printStackTrace", "()V");

    (*env)->CallVoidMethod(env, exception, printStackTraceMethod);
}

现在,在此代码中,本机printStackTraceNative方法可以正常工作并打印堆栈跟踪,但getStackTraceNative不会。当我检查核心转储文件时,它说java已经为java/lang/NoSuchMethodError抛出异常getStackTrace。我很困惑,因为我传递给getStackTraceNative方法的参数是throwable类型,throwable应该有方法getStackTrace

我可能在这里缺少什么概念,对此有任何帮助表示赞赏。谢谢

1 个答案:

答案 0 :(得分:1)

您的方法签名错误。在方法签名中,限定类名称中的句点将由正斜杠替换。因此getStackTrace的签名为()[Ljava/lang/StackTraceElement;(也请注意分号)

您可以使用javap -s选项从类文件中获取方法签名:

javap -classpath '/path/to/jre/lib/rt.jar' -s java.lang.Throwable