我试图用一些例子来理解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
。
我可能在这里缺少什么概念,对此有任何帮助表示赞赏。谢谢
答案 0 :(得分:1)
您的方法签名错误。在方法签名中,限定类名称中的句点将由正斜杠替换。因此getStackTrace
的签名为()[Ljava/lang/StackTraceElement;
(也请注意分号)
您可以使用javap
-s
选项从类文件中获取方法签名:
javap -classpath '/path/to/jre/lib/rt.jar' -s java.lang.Throwable