您的位置:首页 > 编程语言 > Java开发

Java异常学习总结

2013-06-07 08:50 363 查看
一、官方API对此的解释:

1、Throwable:异常和错误的基类,提供了错误堆栈实现等一系列方法。 两个直接子类: Error & Exception

2、两个子类区别:

Error: 程序不应该捕捉的错误,应该交由JVM来处理。一般可能指非常重大的错误。

Exception:程序中应该要捕获的错误。

RuntimeException:运行期异常,是Exception的子类,但勿需捕捉的异常超类

二、各自代码中的表象

Error & RuntimeException:不需要异常处理的,即无try/catch or throws的,编译器认可此种方式;

Exception(排除RuntimeException):编译器强制要求做异常处理的,即必须要 try/catch or throws的;

进一步解释: Exception分为checked Exception & unchecked Exception两种,

unchecked Exception指勿须检查的异常,也即RuntimeException/Error,运行期错误JVM来打理;

checked Exception指必须处理的异常,否则编译器就不让过;

三、为何这样设计?

如上述,Throwable存在2种维度的设计: 1、是否checked 2、 异常级别

是否checked: 表象很明显,编译器会强制检测;

异常级别: 分为 错误和异常,可这个如何来区分呢?我确实很困惑,应该是个人理解的把握了。

四、异常常用关键点知识

1、保留异常源头

异常处理是层级追溯机制,因此异常可被多次抛出,但只要不改变源异常的句柄及其stackTrace,就可一直保留异常源头。

也即简单的throw e(最初的那个异常),异常源头依旧被保留。

2、改变异常源头

如果想改变异常源头,通过抛出一个新异常或在层级机制处理中改变其stackTrace即可。

如在调用方法中e.fillStackTrace() or throw new Excepiton("...")即可定位当前方法为异常源头。

3、自定义异常

一般来说,自定义异常多以checked的为主,也即从Exception继承,实现非常简单,复杂度就看业务需要了,就不多说了。

4、附上部分代码,参照Thinking in java中的实现

public class ThrowableExceptionStudy {
// 异常测试
public static void main(String[] args) throws InterruptedException {
reThrowTest();  // 异常重新抛出测试(可控制改变源头)
rethrowNew();  // 抛出一个新异常测试(改变源头)
runtimeExceptionTest(); // 运行期异常测试
errorTest();  // error测试
}

/**
* 异常重新抛出测试
*/
private static void reThrowTest() {
try {
g();
} catch (Exception e) { // 抛出Exception
System.out.println("Caught exception in reThrowTest, e.printStackTrace()");
e.printStackTrace();
} catch    (Throwable t) { // 抛出Throwable
System.out.println("Caught throwable in reThrowTest, t.printStackTrace()");
t.printStackTrace();
}
}

/**
* 异常源头
*/
private static void f() throws Exception {
System.out.println("originating the exception in f()");
throw new Exception("thrown from f()");
}

/**
* 异常重新抛出,不改变异常句柄
* 以两种方式抛出:1、不改变异常源头 2、改变异常源头
*/
private static void g() throws Throwable {
try {
f();
} catch (Exception e) {
System.out.println("Inside g(), e.printStackTrace()");
e.printStackTrace();
//throw e; // a: 不改变异常源头
throw e.fillInStackTrace(); // b:改变异常源头, 通过fillInStatckTrace实现
}
}

/**
* 抛出一个新的异常,改变了异常句柄
*/
private static void rethrowNew() {
try {
f();
} catch (Exception e) {
System.out.println("Caught in rethrowNew, e.printStackTrace()");
e.printStackTrace();
//throw new NullPointerException("from rethrowNew"); // RuntimeException,编译器自动处理
try {
throw new Exception("from rethrowNew"); // 非RuntimeException,必须捕捉
} catch (Exception e1) {
e1.printStackTrace();
}
}
}

/**
* 运行期异常,勿须捕捉,编译器会自动处理
*/
private static void f1() {
throw new RuntimeException("From f1()");
}

/**
* 运行期异常测试
*/
private static void runtimeExceptionTest() {
try{
f1();
} catch (RuntimeException e){
e.printStackTrace();
}
}

/**
* Error错误
*/
private static void error() {
throw new Error("custome error,not exception. ");
}

/**
* Error测试
*/
private static void errorTest(){
try{
error();
} catch (Error t){
t.printStackTrace();
}
}
}


运行结果:

originating the exception in f()
Inside g(), e.printStackTrace()
java.lang.Exception: thrown from f()
at com.chq.study.ThrowableExceptionStudy.f(ThrowableExceptionStudy.java:40)
at com.chq.study.ThrowableExceptionStudy.g(ThrowableExceptionStudy.java:49)
at com.chq.study.ThrowableExceptionStudy.reThrowTest(ThrowableExceptionStudy.java:25)
at com.chq.study.ThrowableExceptionStudy.main(ThrowableExceptionStudy.java:14)
Caught exception in reThrowTest, e.printStackTrace()
java.lang.Exception: thrown from f()
at com.chq.study.ThrowableExceptionStudy.g(ThrowableExceptionStudy.java:54)
at com.chq.study.ThrowableExceptionStudy.reThrowTest(ThrowableExceptionStudy.java:25)
at com.chq.study.ThrowableExceptionStudy.main(ThrowableExceptionStudy.java:14)
originating the exception in f()
Caught in rethrowNew, e.printStackTrace()
java.lang.Exception: thrown from f()
at com.chq.study.ThrowableExceptionStudy.f(ThrowableExceptionStudy.java:40)
at com.chq.study.ThrowableExceptionStudy.rethrowNew(ThrowableExceptionStudy.java:63)
at com.chq.study.ThrowableExceptionStudy.main(ThrowableExceptionStudy.java:15)
java.lang.Exception: from rethrowNew
at com.chq.study.ThrowableExceptionStudy.rethrowNew(ThrowableExceptionStudy.java:69)
at com.chq.study.ThrowableExceptionStudy.main(ThrowableExceptionStudy.java:15)
java.lang.RuntimeException: From f1()
at com.chq.study.ThrowableExceptionStudy.f1(ThrowableExceptionStudy.java:80)
at com.chq.study.ThrowableExceptionStudy.runtimeExceptionTest(ThrowableExceptionStudy.java:88)
at com.chq.study.ThrowableExceptionStudy.main(ThrowableExceptionStudy.java:16)
java.lang.Error: custome error,not exception.
at com.chq.study.ThrowableExceptionStudy.error(ThrowableExceptionStudy.java:98)
at com.chq.study.ThrowableExceptionStudy.errorTest(ThrowableExceptionStudy.java:106)
at com.chq.study.ThrowableExceptionStudy.main(ThrowableExceptionStudy.java:17)
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: