2

当我开发 android 应用程序时,我想制作一个 CrashReport 类,然后使用它向我的服务器发送报告。

我做了一个CrashHandler类,它实现了UncaughtExceptionHandler,并故意在Activity中做一个NullPoint错误的例子,这个类运行良好,但是在方法uncaughtException中有点问题:</p>

    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        ex.printStackTrace(); //**code line 1**

        ....//some other codes and variables

        StackTraceElement[] elements = ex.getStackTrace(); 
        for (StackTraceElement element : elements) {       // **code line 2**
            errorString = errorString + "<br/>" + element.toString();
    }
    }

代码行 1 将堆栈跟踪打印为:

java.lang.NullPointerException

my.app.ExceptionCaughtActivity$1.onClick(ExceptionCaughtActivity.java:25)
android.view.View.performClick(View.java:2486)
android.view.View$PerformClick.run(View.java:9130)
android.os.Handler.handleCallback(Handler.java:587)
android.os.Handler.dispatchMessage(Handler.java:92)
android.os.Looper.loop(Looper.java:130)
android.app.ActivityThread.main(ActivityThread.java:3703)
java.lang.reflect.Method.invokeNative(Native Method)
java.lang.reflect.Method.invoke(Method.java:507)
com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:841)
com.android.internal.os.ZygoteInit.main(ZygoteInit.java:599)
dalvik.system.NativeStart.main(Native Method)

但代码行只显示:

my.app.ExceptionCaughtActivity$1.onClick(ExceptionCaughtActivity.java:25)
android.view.View.performClick(View.java:2486)
android.view.View$PerformClick.run(View.java:9130)
android.os.Handler.handleCallback(Handler.java:587)
android.os.Handler.dispatchMessage(Handler.java:92)
android.os.Looper.loop(Looper.java:130)
android.app.ActivityThread.main(ActivityThread.java:3703)
java.lang.reflect.Method.invokeNative(Native Method)
java.lang.reflect.Method.invoke(Method.java:507)
com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:841)
com.android.internal.os.ZygoteInit.main(ZygoteInit.java:599)
dalvik.system.NativeStart.main(Native Method)

你注意到它没有显示:

"java.lang.NullPointerException"

我知道我没有调用 getCause(),但是当尝试使用 getCause() 时,它返回 null,而不是 throwable;

我试图查看可抛出类的源代码......但注意到

你能告诉我为什么 getCause 为空但 printStackTrace 有一个错误原因打印出来吗?

4

4 回答 4

1

ex.printStackTrace()有助于打印 Throwable 对象的消息(如果有)ex以及跟踪信息。见Throwable.toString()

原始堆栈跟踪本身不包含此信息,但您可以从 Throwable 对象中获取它,该对象可能有也可能没有消息,可能有也可能没有原因。不过,它总是有一堂课!;-)

这是来自的源代码Throwable

public void printStackTrace(PrintStream s) {
    synchronized (s) {
        s.println(this);
        StackTraceElement[] trace = getOurStackTrace();
        for (int i=0; i < trace.length; i++)
            s.println("\tat " + trace[i]);

        Throwable ourCause = getCause();
        if (ourCause != null)
            ourCause.printStackTraceAsCause(s, trace);
    }
}
于 2012-02-09T13:26:48.567 回答
0

getCause()null如果 位于throwable异常链的顶部,则返回。这是原因,所以它不可能有原因。

您可以在此处阅读有关异常链接的信息。

于 2012-02-09T13:25:22.647 回答
0

您是否尝试过 getMesssage 或 getLocalizedMessage?

http://docs.oracle.com/javase/6/docs/api/java/lang/Throwable.html#getMessage ()

于 2012-02-09T13:29:11.800 回答
0

第一行来自于此:

public String toString() {
    String s = getClass().getName();
    String message = getLocalizedMessage();
    return (message != null) ? (s + ": " + message) : s;
}

所以如果你想添加第一行使用:

errorString += ex.toString + "<br/>";
StackTraceElement[] elements = ex.getStackTrace(); 
for (StackTraceElement element : elements) {       // **code line 2**
    errorString = errorString + "<br/>" + element.toString();
}
于 2012-02-09T13:56:09.157 回答