首页 > 代码库 > [编织消息框架][JAVA核心技术]异常基础
[编织消息框架][JAVA核心技术]异常基础
Java异常体系结构
Thorwable类所有异常和错误的超类,有两个子类Error和Exception,分别表示错误和异常。
其中异常类Exception又分为运行时异常(RuntimeException)和编译时异常(checked Exception),
下面将详细讲述这些异常之间的区别与联系:
1.Error与Exception
Error是程序无法处理的错误,比如OutOfMemoryError、ThreadDeath等。这些异常发生时, Java虚拟机(JVM)一般会选择线程终止。
Exception是程序本身可以处理的异常,这种异常分两大类运行时异常和非运行时异常。 程序中应当尽可能去处理这些异常。
2.运行时异常和编译时异常
运行时异常都是RuntimeException类及其子类异常,如NullPointerException、IndexOutOfBoundsException等,
这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的。
编译时异常是RuntimeException以外的异常,类型上都属于Exception类及其子类。
从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。
3.异常链
设计目的是清晰知道程序出错调用流程,也就是传递性,一般只关心第一个跟最后一个异常信息
源码分析
分三部份:
1.构造异常
2.注册当前异常到异常列表深度坐标
3.打印异常链
第一部份:每次构造异常类时会调用顶级类Throwable构造方法
public Exception(String message) { super(message); } public Exception(String message, Throwable cause) { super(message, cause); }
public Throwable(String message) { fillInStackTrace(); detailMessage = message; } public Throwable(String message, Throwable cause) { fillInStackTrace(); detailMessage = message; this.cause = cause; }
第二部份:注册到异常列表
public synchronized Throwable fillInStackTrace() { if (stackTrace != null || backtrace != null /* Out of protocol state */ ) { //插入异常列表 0 代表插入到第一,也就是说最后一个异常放到头部 fillInStackTrace(0); stackTrace = UNASSIGNED_STACK; } return this; } private native Throwable fillInStackTrace(int dummy);
第三部份:打印异常链,由于最后异常注册到头部,所以循环输出顺序是从尾到头
1 public void printStackTrace() { 2 printStackTrace(System.err); 3 } 4 5 public void printStackTrace(PrintStream s) { 6 printStackTrace(new WrappedPrintStream(s)); 7 } 8 9 private void printStackTrace(PrintStreamOrWriter s) {10 // Guard against malicious overrides of Throwable.equals by11 // using a Set with identity equality semantics.12 Set<Throwable> dejaVu =13 Collections.newSetFromMap(new IdentityHashMap<Throwable, Boolean>());14 dejaVu.add(this);15 16 synchronized (s.lock()) {17 // Print our stack trace18 s.println(this);19 StackTraceElement[] trace = getOurStackTrace();20 for (StackTraceElement traceElement : trace)21 s.println("\tat " + traceElement);22 23 // Print suppressed exceptions, if any24 for (Throwable se : getSuppressed())25 se.printEnclosedStackTrace(s, trace, SUPPRESSED_CAPTION, "\t", dejaVu);26 27 // Print cause, if any28 Throwable ourCause = getCause();29 if (ourCause != null)30 ourCause.printEnclosedStackTrace(s, trace, CAUSE_CAPTION, "", dejaVu);31 }32 }33 34 public StackTraceElement[] getStackTrace() {35 return getOurStackTrace().clone();36 }37 38 private synchronized StackTraceElement[] getOurStackTrace() {39 // Initialize stack trace field with information from40 // backtrace if this is the first call to this method41 if (stackTrace == UNASSIGNED_STACK ||42 (stackTrace == null && backtrace != null) /* Out of protocol state */) {43 int depth = getStackTraceDepth();44 stackTrace = new StackTraceElement[depth];45 for (int i=0; i < depth; i++)46 stackTrace[i] = getStackTraceElement(i);47 } else if (stackTrace == null) {48 return UNASSIGNED_STACK;49 }50 return stackTrace;51 }
4.追踪出错代码
第一个输出是最后一个异常:也就是说可以追踪到最外层出错方法
最后一个输出是第一个异常:也就是说追踪到出错的源头
来个示例说明:
1 package com.eyu.onequeue; 2 3 public class TestException { 4 public static void main(String[] args) { 5 new TestException().demo(); 6 } 7 8 public void a() { 9 try {10 b();11 } catch (Exception e) {12 throw new RuntimeException("call a", e);13 }14 15 }16 17 public void b() {18 try {19 c();20 } catch (Exception e) {21 throw new RuntimeException("call b", e);22 }23 }24 25 public void c() {26 throw new RuntimeException("call c");27 }28 29 public void demo() {30 a();31 }32 }
Exception in thread "main" java.lang.RuntimeException: call a
at com.eyu.onequeue.TestException.a(TestException.java:12)
at com.eyu.onequeue.TestException.demo(TestException.java:30)
at com.eyu.onequeue.TestException.main(TestException.java:5)
Caused by: java.lang.RuntimeException: call b
at com.eyu.onequeue.TestException.b(TestException.java:21)
at com.eyu.onequeue.TestException.a(TestException.java:10)
... 2 more
Caused by: java.lang.RuntimeException: call c
at com.eyu.onequeue.TestException.c(TestException.java:26)
at com.eyu.onequeue.TestException.b(TestException.java:19)
... 3 more
c方法是出错的源头,a方法是调用者
小提示:当异常信息太多时,过滤条件以项目包名或类名能快速定位出错源头,一般是应用程序逻辑出错
下节利用异常开发项目
最后给出jdk常见异常
[编织消息框架][JAVA核心技术]异常基础