解析Java中一切毛病和異常的父類java.lang.Throwable。本站提示廣大學習愛好者:(解析Java中一切毛病和異常的父類java.lang.Throwable)文章只能為提供參考,不一定能成為您想要的結果。以下是解析Java中一切毛病和異常的父類java.lang.Throwable正文
在java說話中,毛病類的基類是java.lang.Error,異常類的基類是java.lang.Exception。
1)雷同點:java.lang.Error和java.lang.Exception都是java.lang.Throwable的子類,是以java.lang.Error和java.lang.Exception本身及其子類都可以作為throw的應用對象,如:throw new MyError();和throw new MyException();個中,MyError類是java.lang.Error的子類,MyException類是java.lang.Exception的子類。
2)分歧點:java.lang.Error本身及其子類不須要try-catch語句的支撐,可在任什麼時候候將前往辦法,以下面的辦法界說:
public String myMethod() { throw new MyError(); }
個中MyError類是java.lang.Error類的子類。
java.lang.Exception本身及其子類須要try-catch語句的支撐,以下的辦法界說是毛病的:
public String myMethod() { throw new MyException(); }
准確的辦法界說以下:
public String myMethod() throws MyException { throw new MyException(); }
個中MyException類是java.lang.Exception的子類。
JAVA異常是在java法式運轉的時刻碰到非正常的情形而創立的對象,它封裝了異常信息,java異常的根類為java.lang.Throwable,全部類有兩個直接子類java.lang.Error和java.lang.Exception.Error是法式自己沒法恢復的嚴重毛病.Exception則表現可以被法式捕捉並處置的異常毛病.JVM用辦法挪用棧來跟蹤每一個線程中一系列的辦法挪用進程,該棧保留了每一個挪用辦法的當地信息.關於自力的JAVA法式,可以一向到該法式的main辦法.當一個新辦法被挪用的時刻,JVM把描寫該辦法的棧構造置入棧頂,位於棧頂的辦法為准確履行的辦法.當一個JAVA辦法正常履行終了,JVM回從挪用棧中彈處該辦法的棧構造,然後持續處置前一個辦法.假如java辦法在履行代碼的進程中拋出異常,JVM必需找到能捕捉異常的catch塊代碼.它起首檢查以後辦法能否存在如許的catch代碼塊,假如存在就履行該 catch代碼塊,不然JVM回挪用棧中彈處該辦法的棧構造,持續到前一個辦法中查找適合的catch代碼塊.最初假如JVM向上追到了main()辦法,也就是一向把異常拋給了main()辦法,依然沒有找到該異常處置的代碼塊,該線程就會異常終止,假如該線程是主線程,運用法式也隨之終止,此時 JVM將把異常直接拋給用戶,在用戶終端上會看到原始的異常信息.
Java.lang.throwable源代碼解析
package java.lang; import java.io.*; /** * * Throwable是一切Error和Exceptiong的父類 * 留意它有四個結構函數: * Throwable() * Throwable(String message) * Throwable(Throwable cause) * Throwable(String message, Throwable cause) * */ public class Throwable implements Serializable { private static final long serialVersionUID = -3042686055658047285L; /** * Native code saves some indication of the stack backtrace in this slot. */ private transient Object backtrace; /** * 描寫此異常的信息 */ private String detailMessage; /** * 表現以後異常由誰人Throwable惹起 * 假如為null表現此異常不是由其他Throwable惹起的 * 假如此對象與本身雷同,注解此異常的原由對象還沒有被初始化 */ private Throwable cause = this; /** * 描寫異常軌跡的數組 */ private StackTraceElement[] stackTrace; /** * 結構函數,原由對象沒有被初始化可以在今後應用initCause停止初始化 * fillInStackTrace可以用來初始化它的異常軌跡的數組 */ public Throwable() { fillInStackTrace(); } /** * 結構函數 */ public Throwable(String message) { //填充異常軌跡數組 fillInStackTrace(); //初始化異常描寫信息 detailMessage = message; } /** * 結構函數,cause表現原由對象 */ public Throwable(String message, Throwable cause) { fillInStackTrace(); detailMessage = message; this.cause = cause; } /** * 結構函數 */ public Throwable(Throwable cause) { fillInStackTrace(); detailMessage = (cause==null ? null : cause.toString()); this.cause = cause; } /** * 獲得具體信息 */ public String getMessage() { return detailMessage; } /** * 獲得具體信息 */ public String getLocalizedMessage() { return getMessage(); } /** * 獲得原由對象 */ public Throwable getCause() { return (cause==this ? null : cause); } /** * 初始化原由對象,這個辦法只能在未被初始化的情形下挪用一次 */ public synchronized Throwable initCause(Throwable cause) { //假如不是未初始化狀況則拋出異常 if (this.cause != this) throw new IllegalStateException("Can't overwrite cause"); //要設置的原由對象與本身相等則拋出異常 if (cause == this) throw new IllegalArgumentException("Self-causation not permitted"); //設置原由對象 this.cause = cause; //前往設置的原由的對象 return this; } /** * 字符串表現情勢 */ public String toString() { String s = getClass().getName(); String message = getLocalizedMessage(); return (message != null) ? (s + ": " + message) : s; } /** * 打印失足誤軌跡 */ public void printStackTrace() { printStackTrace(System.err); } /** * 打印失足誤軌跡 */ public void printStackTrace(PrintStream s) { synchronized (s) { //挪用以後對象的toString辦法 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); } } /** * 打印原由對象的信息 * @param s 打印的流 * @param causedTrace 有此對象惹起的異常的異常軌跡 */ private void printStackTraceAsCause(PrintStream s, StackTraceElement[] causedTrace) { //取得以後的異常軌跡 StackTraceElement[] trace = getOurStackTrace(); //m為以後異常軌跡數組的最初一個元素地位, //n為以後對象惹起的異常的異常軌跡數組的最初一個元素 int m = trace.length-1, n = causedTrace.length-1; //分離從兩個數組的前面做輪回,假如相等則一向輪回,直到不等或數組到頭 while (m >= 0 && n >=0 && trace[m].equals(causedTrace[n])) { m--; n--; } //雷同的個數 int framesInCommon = trace.length - 1 - m; //打印出分歧的毛病軌跡 s.println("Caused by: " + this); for (int i=0; i <= m; i++) s.println("\tat " + trace[i]); //假如有雷同的則打印出雷同的個數 if (framesInCommon != 0) s.println("\t... " + framesInCommon + " more"); //取得此對象的原由對象,並遞歸打印出信息 Throwable ourCause = getCause(); if (ourCause != null) ourCause.printStackTraceAsCause(s, trace); } /** * 打印失足誤軌跡 */ public void printStackTrace(PrintWriter 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); } } /** * 打印原由對象的信息 */ private void printStackTraceAsCause(PrintWriter s, StackTraceElement[] causedTrace) { // assert Thread.holdsLock(s); // Compute number of frames in common between this and caused StackTraceElement[] trace = getOurStackTrace(); int m = trace.length-1, n = causedTrace.length-1; while (m >= 0 && n >=0 && trace[m].equals(causedTrace[n])) { m--; n--; } int framesInCommon = trace.length - 1 - m; s.println("Caused by: " + this); for (int i=0; i <= m; i++) s.println("\tat " + trace[i]); if (framesInCommon != 0) s.println("\t... " + framesInCommon + " more"); // Recurse if we have a cause Throwable ourCause = getCause(); if (ourCause != null) ourCause.printStackTraceAsCause(s, trace); } /** * 填充異常軌跡 */ public synchronized native Throwable fillInStackTrace(); /** * 前往以後的異常軌跡的拷貝 */ public StackTraceElement[] getStackTrace() { return (StackTraceElement[]) getOurStackTrace().clone(); } /** * 獲得以後的異常軌跡 */ private synchronized StackTraceElement[] getOurStackTrace() { //假如第一次挪用此辦法則初始化異常軌跡數組 if (stackTrace == null) { //取得異常軌跡深度 int depth = getStackTraceDepth(); //創立新的異常軌跡數組,並填充它 stackTrace = new StackTraceElement[depth]; for (int i=0; i < depth; i++) stackTrace[i] = getStackTraceElement(i);//獲得指定位標的異常軌跡 } return stackTrace; } /** * 設置異常軌跡 */ public void setStackTrace(StackTraceElement[] stackTrace) { //拷貝設置參數 StackTraceElement[] defensiveCopy = (StackTraceElement[]) stackTrace.clone(); //假如設置參數有空元素則拋出異常 for (int i = 0; i < defensiveCopy.length; i++) if (defensiveCopy[i] == null) throw new NullPointerException("stackTrace[" + i + "]"); //設置以後對象的異常軌跡 this.stackTrace = defensiveCopy; } /** * 異常軌跡的深度,0表現沒法取得 */ private native int getStackTraceDepth(); /** * 獲得指定位標的異常軌跡 */ private native StackTraceElement getStackTraceElement(int index); private synchronized void writeObject(java.io.ObjectOutputStream s) throws IOException { getOurStackTrace(); s.defaultWriteObject(); } }