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

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

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
目录
相关文章
|
1月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
59 1
|
1月前
|
Java API 调度
如何避免 Java 中的 TimeoutException 异常
在Java中,`TimeoutException`通常发生在执行操作超过预设时间时。要避免此异常,可以优化代码逻辑,减少不必要的等待;合理设置超时时间,确保其足够完成正常操作;使用异步处理或线程池管理任务,提高程序响应性。
62 12
|
1月前
|
Java
在 Java 中,如何自定义`NumberFormatException`异常
在Java中,自定义`NumberFormatException`异常可以通过继承`IllegalArgumentException`类并重写其构造方法来实现。自定义异常类可以添加额外的错误信息或行为,以便更精确地处理特定的数字格式转换错误。
35 1
|
1月前
|
IDE 前端开发 Java
怎样避免 Java 中的 NoSuchFieldError 异常
在Java中避免NoSuchFieldError异常的关键在于确保类路径下没有不同版本的类文件冲突,避免反射时使用不存在的字段,以及确保所有依赖库版本兼容。编译和运行时使用的类版本应保持一致。
66 7
|
1月前
|
Java 编译器
如何避免在 Java 中出现 NoSuchElementException 异常
在Java中,`NoSuchElementException`通常发生在使用迭代器、枚举或流等遍历集合时,尝试访问不存在的元素。为了避免该异常,可以在访问前检查是否有下一个元素(如使用`hasNext()`方法),或者使用`Optional`类处理可能为空的情况。正确管理集合边界和条件判断是关键。
71 6
|
1月前
|
Java
Java异常捕捉处理和错误处理
Java异常捕捉处理和错误处理
64 1
|
1月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
60 2
|
1月前
|
Java
如何在 Java 中处理“Broken Pipe”异常
在Java中处理“Broken Pipe”异常,通常发生在网络通信中,如Socket编程时。该异常表示写入操作的另一端已关闭连接。解决方法包括:检查网络连接、设置超时、使用try-catch捕获异常并进行重试或关闭资源。
104 5
|
1月前
|
存储 安全 Java
如何避免 Java 中的“ArrayStoreException”异常
在Java中,ArrayStoreException异常通常发生在尝试将不兼容的对象存储到泛型数组中时。为了避免这种异常,确保在操作数组时遵循以下几点:1. 使用泛型确保类型安全;2. 避免生类型(raw types)的使用;3. 在添加元素前进行类型检查。通过这些方法,可以有效防止 ArrayStoreException 的发生。
41 3
|
2月前
|
人工智能 Oracle Java
解决 Java 打印日志吞异常堆栈的问题
前几天有同学找我查一个空指针问题,Java 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。
43 2
下一篇
DataWorks