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 {
}


目录
相关文章
|
7天前
|
Java 编译器
探索Java中的异常处理机制
【10月更文挑战第35天】在Java的世界中,异常是程序运行过程中不可避免的一部分。本文将通过通俗易懂的语言和生动的比喻,带你了解Java中的异常处理机制,包括异常的类型、如何捕获和处理异常,以及如何在代码中有效地利用异常处理来提升程序的健壮性。让我们一起走进Java的异常世界,学习如何优雅地面对和解决问题吧!
|
7天前
|
Java 开发者
Java中的异常处理:从基础到高级
【10月更文挑战第35天】在Java的世界里,异常处理是维护程序健壮性的关键。本文将深入浅出地探讨Java的异常处理机制,从基本的try-catch语句到自定义异常类的实现,带领读者理解并掌握如何在Java中优雅地处理错误和异常。我们将通过实际代码示例,展示如何捕获、处理以及预防潜在的运行时错误,确保程序即使在面临意外情况时也能保持稳定运行。
22 7
|
6天前
|
Java 数据库连接 开发者
Java中的异常处理机制及其最佳实践####
在本文中,我们将探讨Java编程语言中的异常处理机制。通过深入分析try-catch语句、throws关键字以及自定义异常的创建与使用,我们旨在揭示如何有效地管理和响应程序运行中的错误和异常情况。此外,本文还将讨论一些最佳实践,以帮助开发者编写更加健壮和易于维护的代码。 ####
|
10天前
|
Java
Java 异常处理下篇:11 个异常处理最佳实践
本文深入探讨了 Java 异常处理的最佳实践,包括早抛出晚捕获、只捕获可处理的异常、不要忽略捕获的异常、抛出具体检查性异常、正确包装自定义异常、记录或抛出异常但不同时执行、避免在 `finally` 块中抛出异常、避免使用异常进行流程控制、使用模板方法处理重复的 `try-catch`、尽量只抛出与方法相关的异常以及异常处理后清理资源。通过遵循这些实践,可以提高代码的健壮性和可维护性。
|
11天前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
26 2
|
12天前
|
Java 程序员 数据库连接
深入浅出Java异常处理
【10月更文挑战第30天】在Java的世界里,异常处理就像是生活中的急救箱,遇到意外时能及时救治。本文不仅教你如何使用try-catch语句包扎“伤口”,还会深入讲解如何通过自定义异常来应对那些常见的“头疼脑热”。准备好,我们将一起探索Java异常处理的奥秘,让你的程序更加健壮。
|
13天前
|
Java 程序员 数据库连接
Java中的异常处理:理解与实践
【10月更文挑战第29天】在Java编程的世界里,异常像是不请自来的客人,它们可能在任何时候闯入我们的程序宴会。了解如何妥善处理这些意外访客,不仅能够保持我们程序的优雅和稳健,还能确保它不会因为一个小小的失误而全盘崩溃。本文将通过浅显易懂的方式,带领读者深入异常处理的核心概念,并通过实际示例展现如何在Java代码中实现有效的异常管理策略。
|
16天前
|
Java 数据库连接 数据库
如何构建高效稳定的Java数据库连接池,涵盖连接池配置、并发控制和异常处理等方面
本文介绍了如何构建高效稳定的Java数据库连接池,涵盖连接池配置、并发控制和异常处理等方面。通过合理配置初始连接数、最大连接数和空闲连接超时时间,确保系统性能和稳定性。文章还探讨了同步阻塞、异步回调和信号量等并发控制策略,并提供了异常处理的最佳实践。最后,给出了一个简单的连接池示例代码,并推荐使用成熟的连接池框架(如HikariCP、C3P0)以简化开发。
35 2
|
17天前
|
安全 Java UED
深入理解Java中的异常处理机制
【10月更文挑战第25天】在编程世界中,错误和意外是不可避免的。Java作为一种广泛使用的编程语言,其异常处理机制是确保程序健壮性和可靠性的关键。本文通过浅显易懂的语言和实际示例,引导读者了解Java异常处理的基本概念、分类以及如何有效地使用try-catch-finally语句来处理异常情况。我们将从一个简单的例子开始,逐步深入到异常处理的最佳实践,旨在帮助初学者和有经验的开发者更好地掌握这一重要技能。
19 2
|
18天前
|
Java 程序员 开发者
Java编程中的异常处理艺术
【10月更文挑战第24天】在Java的世界里,代码就像一场精心编排的舞蹈,每一个动作都要精准无误。但就像最完美的舞者也可能踩错一个步伐一样,我们的程序偶尔也会遇到意外——这就是所谓的异常。本文将带你走进Java的异常处理机制,从基本的try-catch语句到高级的异常链追踪,让你学会如何优雅地处理这些不请自来的“客人”。