一文讲解Java中的异常处理机制
Java中的异常处理机制用于处理程序运行过程中可能发生的各种异常情况,通常通过try-catch-finally语句和throw关键字来实现
- Throwable是Java语言中所有错误和异常的父类。它有两个主要的子类:Error和Exception;
- Error类代表那些严重的错误,这类错误通常是程序没法处理的,比如OutOfMemory内存不足,StackOverflowError栈溢出。这些错误通常都和JVM的运行状态有关,一旦发生,应用程序通常就没法恢复;
- Exception类代表可以处理的异常,它分为两大类:编译时异常Checked Exception 和运行时异常Runtime Exception;
- 编译时异常,这类异常在编译时必须被显式处理,捕获或者声明抛出,常见的有IOException、SQLexception等;
- 运行时异常。这类异常在运行时抛出,它们都是RuntimeException的子类。对于运行时异常,Java编译器不要求必须处理它们;运行时异常通常是由程序逻辑错误导致的,比如NullPointerException、IndexOutOfBoundsException 等。
问一个问题,catch和finally的异常可以同时抛出吗?
-
如果catch块抛出一个异常,而finally块中也抛出异常,那么最终抛出的异常将是finally块中的异常。catch块中的异常会被丢弃,而finally块中的异常会覆盖并向上传递;
public class Example { public static void main(String[] args) { try { throw new Exception("Exception in try"); } catch (Exception e) { throw new RuntimeException("Exception in catch"); } finally { throw new IllegalArgumentException("Exception in finally"); } } }
- try 块首先抛出一个 Exception。
- 控制流进入 catch 块,catch 块中又抛出了一个 RuntimeException。
- 但是在 finally 块中,抛出了一个 IllegalArgumentException,最终程序抛出的异常是 finally 块中的 IllegalArgumentException。
虽然 catch 和 finally 中的异常不能同时抛出,但可以手动捕获 finally 块中的异常,并将 catch 块中的异常保留下来,避免被覆盖。常见的做法是使用一个变量临时存储 catch 中的异常,然后在 finally 中处理该异常:
public class Example { public static void main(String[] args) { Exception catchException = null; try { throw new Exception("Exception in try"); } catch (Exception e) { catchException = e; throw new RuntimeException("Exception in catch"); } finally { try { throw new IllegalArgumentException("Exception in finally"); } catch (IllegalArgumentException e) { if (catchException != null) { System.out.println("Catch exception: " + catchException.getMessage()); } System.out.println("Finally exception: " + e.getMessage()); } } } }