在 Java 中,异常可以分为两大类:Checked异常(已检查异常) 和 运行时异常(Unchecked异常)。它们的主要区别在于何时被检测到、如何处理,以及它们的继承层次结构。以下是它们的详细区别:
1. 定义与继承层次
-
Checked异常(已检查异常):
- 定义:Checked异常是编译时被检测的异常,必须要进行处理(即要么捕获异常,要么在方法签名中声明抛出)。
- 继承层次:Checked异常是
Exception
类及其子类,但不包括RuntimeException
及其子类。 - 示例:
IOException
、SQLException
、ClassNotFoundException
。
-
运行时异常(Unchecked异常):
- 定义:运行时异常是在程序运行时可能发生的异常,编译器不会强制要求进行处理,可以选择不处理。
- 继承层次:运行时异常是
RuntimeException
类及其子类。 - 示例:
NullPointerException
、ArrayIndexOutOfBoundsException
、ArithmeticException
。
2. 处理要求
-
Checked异常:
- 必须处理:编译器要求必须对 Checked异常进行处理。处理方式可以是用
try-catch
块捕获,或者在方法签名中使用throws
关键字声明抛出该异常。 - 编译时检查:如果没有处理或声明抛出 Checked异常,程序在编译时会报错。
public void readFile(String fileName) throws IOException { // 文件操作可能抛出 IOException,必须处理 FileReader file = new FileReader(fileName); }
- 必须处理:编译器要求必须对 Checked异常进行处理。处理方式可以是用
-
运行时异常:
- 不强制处理:编译器不会强制要求对运行时异常进行处理。开发者可以选择处理这些异常,但如果不处理,程序运行时可能会因为这些异常而崩溃。
- 运行时检查:运行时异常是程序运行时才会抛出的,因此它们通常是由编程错误或未预见的情况引起的。
public void divide(int a, int b) {
int result = a / b; // 如果 b 为 0,会抛出 ArithmeticException
}
3. 典型用例
-
Checked异常:
- 主要用于表示外部因素导致的错误,如文件未找到、网络故障、数据库连接失败等。这些异常通常是可以预见的,并且需要处理以确保程序的稳定性。
- 示例:文件操作时可能抛出的
IOException
,数据库操作时可能抛出的SQLException
。
-
运行时异常:
- 主要用于表示程序逻辑中的错误,如数组越界、空指针访问、非法参数等。这些异常通常是由于编写不当的代码或未考虑到的边界情况引起的。
- 示例:访问空对象时抛出的
NullPointerException
,数组访问越界时抛出的ArrayIndexOutOfBoundsException
。
4. 开发中的实践
-
Checked异常:
- 在实际开发中,建议对可能发生的 Checked异常进行处理,以增强程序的健壮性和容错性。例如,使用
try-catch
捕获异常并记录日志或提供友好的用户提示。
- 在实际开发中,建议对可能发生的 Checked异常进行处理,以增强程序的健壮性和容错性。例如,使用
-
运行时异常:
- 由于运行时异常通常是编程错误导致的,开发中更应该关注如何避免这些错误。例如,提前检查参数的合法性或在使用对象前确保它们已经被正确初始化。对于不可避免的运行时异常,可以选择捕获并进行适当处理,避免程序崩溃。
总结
- Checked异常:编译器要求必须处理,通常与外部资源或环境相关的异常。
- 运行时异常:编译器不强制要求处理,通常与程序逻辑或编写错误相关的异常。
通过理解这两类异常的区别,开发者可以更好地编写健壮且可靠的 Java 应用程序。