Java进阶之异常捕捉处理和错误处理

简介: 【7月更文挑战第9天】Java异常处理确保程序在遇到错误时不会崩溃。关键机制包括try-catch-finally,用于捕获(try)、处理(catch)和清理(finally)异常。异常分为检查型(需编译时处理,如IOException)和非检查型(如NullPointerException)。throw用于抛出异常,throws用于声明方法可能抛出的异常。Error表示系统级错误,不可恢复;Exception是可处理的异常,包括检查型和非检查型。自定义异常通过继承Exception实现。Java 7引入try-with-resources自动关闭资源。

Java进阶之异常捕捉处理和错误处理
在代码运行过程中,如果出现错误和异常情况,代码就会崩溃中断执行!
但是我们在代码运行时并不希望代码中断,比如基础总结时候,我们要求输入的时间格式是2024-02-02,并且根据这个预定义格式进行格式化时间。然而用户输入时候非要不按照这个格式来,这样就会导致代码运行发生异常情况,而导致系统崩溃!所以我们需要一套防御机制来避免这种异常情况的发生!
今天我们就来看一下Java中内置的的异常捕捉处理和错误处理!
在Java中,异常处理依赖于以下五个核心关键字:try、catch、finally、throw 和 throws。
语法:
try {
// 运行的业务代码
} catch (Exception e) {
// 只有发生上面()里的异常,才会执行的代码。用来处理异常,
System.out.println("捕获到异常: " + e.getMessage());
// 一般我们要在这里记录详细的错误信息,有助于调试和问题追踪。
// 也可以再次抛出 比如 throw new MyException("新异常信息", e);
}finally{
// 始终会执行的代码
System.out.println("关闭连接");
}
这个结构是标准的异常捕捉语法:
try块:用于包围可能抛出异常的代码。一个try块至少需要搭配一个catch块或finally块。
catch块:用于捕获并处理try块内抛出的异常。可以定义多个catch块来处理不同类型的异常。
finally块:无论try块中的代码是否抛出异常,finally块都会被执行。常用于清理资源,如关闭文件或网络连接。可以省略。

  异常类型
  Java中的异常分为检查型异常(Checked Exceptions)和非检查型异常(Unchecked Exceptions)。
  检查型异常(Checked Exceptions)
    在编译时必须被显式处理的异常。通常是由外部因素导致的,例如文件不存在、网络连接失败等。编译器会强制程序员处理这些异常,要么通过try-catch语句捕获,要么在方法签名中使用throws关键字声明。
    常见的检查型异常包括:
    IOException:输入输出异常,如文件读写错误。
    SQLException:数据库访问异常。
    ClassNotFoundException:类找不到异常。
    NoSuchMethodException:方法找不到异常。
    InvocationTargetException:反射调用目标异常。

  非检查型异常(Unchecked Exceptions)
    包括运行时异常(RuntimeExceptions)和错误(Errors)。这些异常在编译时不需要被显式处理,因为它们通常是由程序逻辑错误或系统错误引起的,也就是说这些就是你写的bug,应该在代码编写阶段避免掉。
    常见的非检查型异常包括:
    NullPointerException:空指针异常。
    ArrayIndexOutOfBoundsException:数组越界异常。
    IllegalArgumentException:非法参数异常。
    ArithmeticException:算术异常,如除以零。
    ClassCastException:类型转换异常。

  检查型异常和非检查型异常区别:
  编译时检查:检查型异常必须在编译时被处理,而非检查型异常则不需要。
  处理要求:检查型异常强制程序员处理,而非检查型异常则不强制,但仍然建议处理。
  异常类型:检查型异常通常是由外部因素导致的,而非检查型异常通常是由程序逻辑错误或系统错误引起的。
  传播方式:检查型异常会强制传递给调用者,而非检查型异常则可以选择性地处理或不处理。

  抛出异常有两种:
  throw关键字:(抛出异常这个动作)
  用于在代码的某个点显式抛出一个异常。
  通常是在方法内部,当发生了一个错误条件,需要停止当前方法的执行,并将错误信息传递给调用者。
  throw后面跟的是一个异常实例。
  一旦抛出了异常,当前方法的执行会立即停止,除非异常被捕获处理。
  throw可以抛出任意类型的Throwable子类实例,包括Error和Exception。
  示例:
  public void validate(int age) {
      if (age < 0) {
          throw new IllegalArgumentException("年龄不能为负数");
          // 到这就停了,不往下执行了
      }
      // 其他代码
  }
  throws关键字:(只是声明可能抛出,不是动作)
  用于声明一个方法可能抛出的异常。
  throws出现在方法签名中,位于参数列表之后。
  使用throws时,可以声明方法可能抛出的一个或多个异常类型。
  throws主要用于检查型异常,即那些不是RuntimeException的异常。
  声明异常后,调用该方法的方法必须处理这些异常,要么通过try-catch捕获,要么继续使用throws声明。
  示例:
  public void readFile(String filename) throws IOException {
      // 可能抛出IOException的代码
  }

  在Java中的内置的异常类以Throwable类为根。Throwable是所有错误和异常的超(父)类。
  Throwable类有两个直接子类:Error和Exception。这个层次结构的主要目的是区分不可恢复的错误和可抛出的异常。
  Error
  Error类表示编译时和系统错误(外部错误),这些错误发生时,Java运行时环境(JRE)通常无法做什么恢复工作。例如,OutOfMemoryError表示JVM没有足够的内存来分配对象,StackOverflowError表示栈溢出。这些错误一般是由外部环境或资源限制导致的,应用程序不应该尝试捕获这些错误。
  RuntimeException
  RuntimeException及其子类表示编程错误或逻辑错误,如尝试除以零(ArithmeticException)、访问空对象的成员(NullPointerException)或数组访问越界(ArrayIndexOutOfBoundsException)。这些异常是未检查的,因为它们通常是程序员可以避免的。

  其他检查型异常
  除了RuntimeException之外的其他Exception子类都是检查型异常。这些异常通常是由外部因素导致的,如文件不存在(FileNotFoundException)、网络问题(SocketException)或数据库错误(SQLException)。编译器强制程序员处理这些异常,要么通过try-catch语句捕获,要么在方法签名中使用throws关键字声明。

  所以Exception我们尚且可以抛出捕捉处理,要是Error的话基本上就是系统出错了!

  以下是Java异常类层次结构的一个简化视图:
  Throwable
  │
  ├── Error
  │   ├── VirtualMachineError
  │   │   ├── StackOverflowError
  │   │   ├── OutOfMemoryError
  │   │   └── ...
  │   ├── AssertionError
  │   └── ...
  │
  └── Exception
      ├── RuntimeException
      │   ├── NullPointerException
      │   ├── ArrayIndexOutOfBoundsException
      │   ├── ArithmeticException
      │   ├── IllegalArgumentException
      │   ├── ClassCastException
      │   └── ...
      ├── IOException
      │   ├── FileNotFoundException
      │   ├── EOFException
      │   ├── SocketException
      │   └── ...
      ├── SQLException
      ├── TimeoutException
      ├── InterruptedException
      ├── ClassNotFoundException
      ├── NoSuchMethodException
      └── ...
  自定义异常
  通过继承Exception类或其子类,可以创建自定义异常,以更好地描述特定场景。
  // 自定义异常类,继承自Exception类
  public class CustomException extends Exception {

      // 构造函数,允许设置异常消息
      public CustomException(String message) {
          super(message);
      }

      // 构造函数,允许设置异常消息和原因
      public CustomException(String message, Throwable cause) {
          super(message, cause);
      }

      // 构造函数,允许设置原因
      public CustomException(Throwable cause) {
          super(cause);
      }
  }

  // 使用自定义异常的示例
  public class CustomExceptionDemo {
      public static void main(String[] args) {
          try {
              // 假设某个条件不满足,抛出自定义异常
              if (someCondition()) {
                  throw new CustomException("这是一个自定义异常");
              }
          } catch (CustomException e) {
              // 捕获并处理自定义异常
              System.out.println("捕获到自定义异常: " + e.getMessage());
          }
      }

      // 一个假设的方法,用于检查某个条件
      public static boolean someCondition() {
          // 这里应该是检查条件逻辑
          return true;
      }
  }
  如果Java内置的异常类不满足你的需求,或者你要针对你得异常分类,就可以自定义一个异常来实现。


  我们在编写代码时候,这些异常类要注意类之间的关系,比如Exception是RuntimeException的父类,如一下写法:
  try {
      // 运行的业务代码
  } catch (Exception e) {
    // 捕捉到了Exception
  } catch (RuntimeException e) {
    // 永远不会执行这里,因为上面Exception已经捕捉到异常了,所有异常都继承自Exception嘛
  }
  还有,如果这么编写:
  public void run(){
    try {
      // 运行的业务代码
    } catch (IOException e) {
      // 捕捉到了Exception
    }
  }
  但是业务代码报了SQLException,那么此时是捕捉不到SQLException异常的,此时默认就是向外抛出,相当于是throws SQLException了:
  public void run() throws SQLException {
    try {
      // 运行的业务代码
    } catch (IOException e) {
      // 捕捉到了Exception
    }
  }
  如果异常抛出后,在上层的调用方法中并没有被捕捉到,那么就会一路向上抛出到main方法,直接中断程序运行。
  如果异常抛出后,在上层调用方法中捕捉到后,可以抛出一个新的异常,并将原始异常传递给新异常,一路向上抛形成异常链。最后正确处理,保证程序稳定运行。

  try-with-resources
  Java 7 引入了一种新的异常处理机制,可以自动关闭实现了AutoCloseable接口的资源。
  try (Resource resource = new Resource()) {
      // 使用资源
  } catch (Exception e) {
      // 处理异常
  }
  只是一种针对实现了AutoCloseable接口的特殊写法额,可以自动关闭,无需去写finally去手动关闭而已。
  通过合理地使用这些机制,可以编写出健壮、可靠的Java应用程序,能够有效地处理运行时出现的各种问题。
  END
目录
相关文章
|
23天前
|
Java 编译器 数据库连接
Java——异常
在 Java 中,程序执行过程中的不正常行为被称为异常。异常分为 Error 和 Exception。Error 表示系统级错误,而 Exception 则封装程序可能出现的问题。异常包括编译时异常和运行时异常(如数组越界)。异常可用于查找 bug 信息和作为方法内部的特殊返回值。处理异常的方式有默认处理和捕获异常,后者通过 try-catch 结构实现。此外,还可以自定义异常类来更灵活地处理特定情况。
28 9
Java——异常
|
26天前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
13天前
|
Java 编译器 索引
|
14天前
|
IDE Java 开发工具
java自定义异常20
java自定义异常20
17 3
|
14天前
|
IDE Java 开发工具
java捕获异常19
java捕获异常19
14 2
|
2月前
|
Java 数据库连接 程序员
Java 认识异常
Java 认识异常
14 1
|
1月前
|
Java C++
Java内存区域于内存溢出异常
这篇文章详细解释了Java虚拟机的内存区域划分、各区域的作用以及可能遇到的内存溢出异常情况。
37 0
Java基础异常-自定义异常
Java基础异常-自定义异常
Java基础异常-自定义异常
|
Java 编译器
Java中的异常(抛出异常、自定义异常等)
Java中的异常(抛出异常、自定义异常等)
284 1
|
Java 程序员 编译器
Java异常——throw、throws及自定义异常
Java异常——throw、throws及自定义异常
124 0