Java 中的异常处理

简介: 程序中可能会有很多意想不到的问题的出现,这些问题中,有些是在编写阶段时就无法编译通过,比如写代码时变量名写错,出现语法错误 java.lang.Error: Unresolved compilation problem ……;有些是在程序运行的时候出现的,比如一个除法程序,结果用户输入的除数是 0,那么就会引发 java.lang.ArithmeticException 等等。异常发生的原因很多,但无论怎样,这些异常和其他的对象一样,都只是类的实例。

一、认识异常

程序中可能会有很多意想不到的问题的出现,这些问题中,有些是在编写阶段时就无法编译通过,比如写代码时变量名写错,出现语法错误 java.lang.Error: Unresolved compilation problem ……;有些是在程序运行的时候出现的,比如一个除法程序,结果用户输入的除数是 0,那么就会引发 java.lang.ArithmeticException 等等。异常发生的原因很多,但无论怎样,这些异常和其他的对象一样,都只是类的实例。

1.1 异常分类

Java 中异常处理主要有三类:

  • 检查性异常(Checked Exception):检查一词可以理解为编译器对代码的检查,如果代码中含有此类异常且没有进行处理,编译就无法通过,比如文件无法找到的异常等。这种异常可以说是 Java 的特色,其他流行语言少有这种异常。这样的异常一般是由程序运行的环境所导致的,程序员无法知道程序最终的运行环境,所以应该时刻准备好应对这些异常(用 try ... catch ... 捕获并处理);
  • 运行时异常(Runtime Exception):这是编译器无法检查到,程序运行时才会出现的异常,比如除数为零这样的异常。对于这类异常,程序员应该先尝试直接修改代码的逻辑避免出现此类异常,实在不行时再用 try ... catch ... 去捕获它并进行处理。一个好的代码,应该尽可能少地抛出运行时异常;
  • 错误(Error):错误是 Java 程序无法处理的严重故障,程序员一般不应该去捕获它们,捕获了可能也无法处理,这也是程序员无法预见的,在编译时编译器也无法检查到,大部分错误都与程序员执行的操作无关,而是虚拟机本身在平台上的运行出现了问题。

下面是异常机制的继承关系图:

异常继承关系图

1.2 内置异常

Java 语言内置的标准异常类型都在 java.lang 标准包中。

1.2.1 运行时异常

下面只列出部分常用的运行时异常:

异常类型名 描述
ArithmeticException 当出现异常的运算条件时(如除数为零),抛出此异常
ArrayIndexOutOfBoundsException 用非法索引(索引为负或大于等于数组大小)访问数组时,抛出此异常
ArrayStoreException 试图将错误类型的对象存储到一个对象数组时,抛出此异常
ClassCastException 当试图将对象强制转换为不是实例的子类时,抛出此异常
IllegalArgumentException 向方法传递了一个不合法或不正确的参数时,抛出此异常
IllegalMonitorStateException 抛出的异常表明某一线程已经试图等待对象的监视器,或者试图通知其他正在等待对象的监视器而本身没有指定监视器的线程
IllegalStateException 在非法或不适当的时间调用方法时产生的信号。换句话说,即 Java 环境或 Java 应用程序没有处于请求操作所要求的适当状态下
IllegalThreadStateException 线程没有处于请求操作所要求的适当状态时抛出的异常
IndexOutOfBoundsException 指示某排序索引(例如对数组、字符串或向量的排序)超出范围时抛出
NegativeArraySizeException 如果应用程序试图创建大小为负的数组,则抛出该异常
NullPointerException 当应用程序试图在需要对象的地方使用 null 时,抛出该异常
NumberFormatException 当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出该异常
SecurityException 由安全管理器抛出的异常,指示存在安全侵犯
StringIndexOutOfBoundsException 此异常由 String 方法抛出,指示索引或者为负,或者超出字符串的大小
UnsupportedOperationException 当不支持请求的操作时,抛出此异常

1.2.2 检查性异常

下面只列出部分检查性异常:

异常类型名 描述
ClassNotFoundException 应用程序试图加载类时,找不到相应的类,抛出该异常
CloneNotSupportedException 当调用 Object 类中的 clone 方法克隆对象,但该对象的类无法实现 Cloneable 接口时,抛出该异常
IllegalAccessException 拒绝访问一个类的时候,抛出该异常
InstantiationException 当试图使用 Class 类中的 newInstance 方法创建一个类的实例,而指定的类对象因为是一个接口或是一个抽象类而无法实例化时,抛出该异常
InterruptedException 一个线程被另一个线程中断,抛出该异常
NoSuchFieldException 请求的变量不存在
NoSuchMethodException 请求的方法不存在

1.2.3 错误

下面只列出部分错误类型:

错误类型名 描述
LinkageError 动态链接失败
VirtualMachineError 虚拟机错误
AWTError AWT错误

1.3 异常的方法

这里只说明 Throwable 的方法,其子类异常还有其他自己的方法,这里不展开讲述。下面是 Throwable 类的主要方法:

方法名 描述
public String getMessage() 返回关于发生的异常的详细信息
public Throwable getCause() 返回一个 Throwable 对象代表异常原因
public String toString() 返回此 Throwable 的简短描述
public void printStackTrace() 将此 Throwable 及其回溯打印到标准错误流
public StackTraceElement [] getStackTrace() 返回一个包含堆栈层次的数组,下标为 0 的元素代表栈顶,最后一个元素代表方法调用堆栈的栈底
public Throwable fillInStackTrace() 用当前的调用栈层次填充 Throwable 对象栈层次,添加到栈层次任何先前信息中

二、捕获异常

2.1 try ... catch ... 语句

和 C++ 语言类似,使用 try 关键字和 catch 关键字捕获异常。在 try 语句块中的部分代码是可能出现异常的代码,catch 关键字后面表示捕获哪一种特殊的异常,并在其语句块中进行相应的处理。

try {
// 某种异常} catch (ExceptionTypee) { // ExceptionType表示某种异常类型,e是这个异常实例对象的引用名// 处理方式}

下面是一个使用 try ... catch ... 的示例:

publicclassTest {
publicstaticvoidmain(String[] args) {
System.out.println(divide(1, 0));
// Output:// 除数不能为零!// 0    }
publicstaticintdivide(intnum1, intnum2) {
try {
returnnum1/num2;
        } catch (ArithmeticExceptione) {
System.out.println("除数不能为零!");
return0;
        }
    }
}

2.2 多重捕获

当异常类型不只一种的时候,就需要多重捕获。

try {
// 某种异常} catch (ExceptionType1e1) {
// 对于第一种异常的处理方式} catch (ExceptionType2e2) {
// 对于第二种异常的处理方式}

2.3 finally 关键字

这个 finally 关键字和 Python 异常处理中的 finally 关键字类似,都是在 try ... catch ... 语句执行完后执行 finally 关键字中的内容。

try {
// 捕获异常} catch (ExceptionTypee) {
// 处理异常} finally {
// 最后执行(不管上面是否对异常进行了处理)}

下面和 Python 的异常处理进行对比:

""" Python 的异常处理 """try:
    ...  # 捕获异常exceptExceptionTypease:  # e是异常类型的实例对象的引用变量名    ...  # 处理异常else:
    ...  # 没有异常时执行的代码finally:
    ...  # 最后必定执行的代码

除了 else 是 Python 中特有的用法之外,其余部分和 Java 中的几乎完全相同。

另外,标准 C++ 的异常处理中没有 finally 关键字(MSVC 中有个 __finally 关键字做了 finally 的替代,但其他的 C++ 编译器不一定有)

总结一下,下面是 try ... catch ... finally ... 语句的执行流程图:

try ... catch ... finally ... 语句

此外,还要注意以下几点问题:

  • catch 或者 finally 关键字不能独立于 try 关键字而存在;
  • 当不是 try-with-resources 语法时,try 后面必须接上 catch 或者 finally 关键字;
  • 多重捕获时,具体异常要放在宽泛异常之前,不然无法捕获到(逻辑上宽泛异常包含了具体异常);

2.4 try-with-resources 语法糖

try-with-resouces 语法糖是 Java 的一种特殊语法,是在 Java7 为了简化语法(方便打开资源)而引入的。在 Java7 之前,打开文件、套接字等,都需要程序员手动地去关闭这些资源,很麻烦,用 finally 也不方便。于是就在 Java7 出现了这个语法糖,它可以在执行完相关的操作后自动关闭这些资源。

该语法糖的基本格式如下:

try (resource_declaration_1; resource_declaration_2; ...) { // 圆括号内可声明或实例化一个或多个(用分号间隔)资源对象// 使用的资源} // 执行完后,资源对象将被自动关闭,关闭顺序与声明或实例化顺序相反

下面是一个示例:

publicstaticvoidtry_with_res() throwsIOException { // 下面产生的异常(被声明了)会传递到方法外部,相当于处理了(后面会讲)try (Scanners=newScanner(newFile("input.txt")); PrintWriterw=newPrintWriter(newFile("output.txt"))) {
while (s.hasNext())
w.print(s.nextLine());
    }
}

其实这个语法糖在资源打开操作方面和 Python 中的 with 关键字有异曲同工之妙,都可以简化资源的操作(打开后自动关闭),下面给出一个示例(Python3.8)以进行对比:

withopen('input.txt', 'r') asinput_file, open('output.txt', 'w') asoutput_file:
whileline :=input_file.readline():
output_file.write(line)

三、抛出异常和声明异常

3.1 throw 关键字

程序员可以用 throw 关键字来显式地抛出异常。

thrownewExceptionObject(...);

下面是一个示例:

publicstaticvoidCheckArgument(Objectarg) {
if (arginstanceofInteger) {
thrownewIllegalArgumentException("参数类型不能为整型!");
    }
}

3.2 throws 关键字

程序员可以用 throws 关键字来为方法声明可能抛出的异常,这只是一个声明,并非运行时一定抛出该异常。当该方法中真的抛出声明中指定的异常时,此异常就会向外传递到方法外部,也就是调用该方法的地方。throws 关键字放在方法头之后,花括号之前,后面跟要声明的异常,可以有多个,通过逗号间隔。

下面是一个示例:

publicstaticvoidCheckArgument(Objectarg) throwsIllegalArgumentException, UnknownError {
if (arginstanceofInteger) {
thrownewIllegalArgumentException("参数类型不能为整型!");
    }
}

四、自定义异常

程序员除了可以使用 Java 内置的异常之外,还可以自己定义一个异常来使用,但无论怎样,自定义的异常也是异常最顶层父类 Throwable 的子类。用途的不同的异常,需要继承的父类异常也不一样,若是要自定义运行时异常,则应该继承类 RuntimeException;若是检查性异常,则应该继承类 Exception;若是错误,则应该继承类 Error。

自定义的异常类型和其他的异常一样,实际都是对象。自定义的异常一般应该满足以下条件,当然,这不是强制的:

  • 有一个无参构造函数;
  • 有一个带有 String 参数的构造函数,并传递给父类的构造函数;
  • 有一个带有 Throwable 参数的构造函数,并传递给父类的构造函数;
  • 有一个带有 String 参数和 Throwable 参数,并都传递给父类构造函数;

下面是一个自定义异常的简单示例:

classZeroDivisionExceptionextendsArithmeticException {
}


目录
相关文章
|
21天前
|
Java 开发者
Java中的异常处理:从基础到高级
在Java编程的世界里,异常处理是一块基石,它确保了程序的健壮性和稳定性。本文将带你从异常的基础概念出发,逐步深入到高级处理技巧,通过实例展示如何在Java中有效管理和处理异常。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
20天前
|
Java 程序员
Java编程中的异常处理:从基础到高级
在Java的世界中,异常处理是代码健壮性的守护神。本文将带你从异常的基本概念出发,逐步深入到高级用法,探索如何优雅地处理程序中的错误和异常情况。通过实际案例,我们将一起学习如何编写更可靠、更易于维护的Java代码。准备好了吗?让我们一起踏上这段旅程,解锁Java异常处理的秘密!
|
17天前
|
Java
Java 异常处理:11 个异常处理最佳实践
本文深入探讨了Java异常处理的最佳实践,包括早抛出晚捕获、只捕获可处理异常、不忽略异常、抛出具体异常、正确包装异常、记录或抛出异常但不同时执行、不在finally中抛出异常、避免用异常控制流程、使用模板方法减少重复代码、抛出与方法相关的异常及异常处理后清理资源等内容,旨在提升代码质量和可维护性。
|
20天前
|
安全 Java 数据库连接
Java中的异常处理:理解与实践
在Java的世界里,异常处理是维护代码健壮性的守门人。本文将带你深入理解Java的异常机制,通过直观的例子展示如何优雅地处理错误和异常。我们将从基本的try-catch结构出发,探索更复杂的finally块、自定义异常类以及throw关键字的使用。文章旨在通过深入浅出的方式,帮助你构建一个更加稳定和可靠的应用程序。
30 5
|
18天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
|
19天前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
43 2
|
19天前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
39 2
|
19天前
|
Java 开发者
Java 中的异常处理:不仅仅是 try-catch
在Java的世界里,异常处理是代码的守护神,它保护着程序不会因为意外错误而崩溃。但异常处理远不止try-catch那么简单。本文将深入探讨Java的异常处理机制,从基本的try-catch到更复杂的自定义异常和finally块的使用,带你理解如何在Java中优雅地处理错误。
50 1
|
22天前
|
Java 程序员 UED
深入理解Java中的异常处理机制
本文旨在揭示Java异常处理的奥秘,从基础概念到高级应用,逐步引导读者掌握如何优雅地管理程序中的错误。我们将探讨异常类型、捕获流程,以及如何在代码中有效利用try-catch语句。通过实例分析,我们将展示异常处理在提升代码质量方面的关键作用。
31 3
|
22天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####