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
目录
相关文章
|
11天前
|
网络协议 Java 编译器
Java常见异常及对应解决办法
Java常见异常及对应解决办法
29 10
|
4天前
|
存储 Java 程序员
|
10天前
|
Java 编译器 程序员
Java面试题-异常
Java面试题-异常
25 6
|
11天前
|
网络协议 Java 数据库连接
13 Java异常(异常过程解析、throw、throws、try-catch关键字)
13 Java异常(异常过程解析、throw、throws、try-catch关键字)
32 2
|
16天前
|
存储 Java 编译器
Java内存区域与内存溢出异常 - 运行时数据区
【8月更文挑战第2天】Java运行时数据区包括:1) 程序计数器:记录线程执行字节码的行号,线程私有;2) Java虚拟机栈:描述方法执行的内存模型,线程私有,深度过大抛出`StackOverflowError`;3) 本地方法栈:服务于Native方法,线程私有;4) Java堆:所有线程共享,对象实例在此分配内存;5) 方法区:存储类信息、常量等数据;6) 运行时常量池:方法区的一部分,存放字面量和符号引用。不当使用如无限创建对象或过度递归调用会导致各种内存溢出错误。
|
24天前
|
Java Perl
Java进阶之正则表达式
【7月更文挑战第17天】正则表达式(RegEx)是一种模式匹配工具,用于在字符串中执行搜索、替换等操作。它由普通字符和特殊元字符组成,后者定义匹配规则。
17 4
|
27天前
|
设计模式 Java
Java进阶之代理
Java进阶之代理
17 4
|
27天前
|
设计模式 Java
Java进阶之代理
Java进阶之代理
16 3
|
27天前
|
设计模式 Java
Java进阶之代理
【7月更文挑战第16天】Java动态代理通过`java.lang.reflect.Proxy`和`InvocationHandler`实现,无需编译期定义代理类。与静态代理相比,它更灵活,代码更简洁,适用于方法数量变化或未知接口代理。
15 2
|
28天前
|
Java 编译器 API
Java进阶之标准注解
【7月更文挑战第15天】Java标准注解包括标记注解(如@Deprecated)、@Override(检查方法重写)、@SuppressWarnings(抑制警告)。多值注解如@RequestMapping在Spring中用于HTTP请求映射。元注解如@Retention控制注解保留策略,@Target指定应用位置。Java8引入类型注解(@FunctionalInterface、@SafeVarargs)和重复注解(@Repeatable)。自定义注解可通过反射读取,如示例中的MyMarkerAnnotation等。
17 2