14 Error 与 Exception —— 异常分类与处理策略

Error 与 Exception ------ 异常分类与处理策略

适用版本: JDK 8 难度等级: 基础 核心概念: 受检/非受检异常、Error 体系、异常设计原则


一、Java 异常体系全景图

javascript 复制代码
                       Throwable
                           │
            ┌──────────────┴──────────────┐
            │                             │
          Error                       Exception
            │                             │
    ┌───────┼───────┐              ┌──────┴──────┐
    │       │       │              │             │
OutOfMemory  ...  StackOverflow  RuntimeException  IOException ...
  Error            Error          │
                           ┌──────┼──────┐
                           │      │      │
                      NullPointer ...  IllegalArgumentException
类型 含义 处理策略
Error JVM 层面的严重问题 不应捕获,应让程序终止
RuntimeException 编程错误(非受检) 修复代码而非捕获
受检异常 可预见的业务异常 必须 try-catch 或 throws

二、Error 类

java 复制代码
public class Error extends Throwable {
    public Error()                                    { super(); }
    public Error(String message)                      { super(message); }
    public Error(String message, Throwable cause)     { super(message, cause); }
    public Error(Throwable cause)                     { super(cause); }
}

常见 Error 子类

Error 类型 含义 典型场景
OutOfMemoryError 堆内存耗尽 内存泄漏、超大对象
StackOverflowError 栈深度超限 无限递归
NoClassDefFoundError 运行时找不到类 编译后 jar 被移除
VirtualMachineError JVM 内部错误 极少遇到
java 复制代码
public class ErrorDemo {
    public static void main(String[] args) {
        // StackOverflowError
        try {
            recursive(0);
        } catch (StackOverflowError e) {
            System.out.println("栈溢出!深度过大");
        }

        // 模拟 OutOfMemoryError
        // byte[][] arrays = new byte[Integer.MAX_VALUE][];
    }

    static void recursive(int depth) {
        recursive(depth + 1);
    }
}

三、Exception 类

java 复制代码
public class Exception extends Throwable {
    public Exception()                                 { super(); }
    public Exception(String message)                   { super(message); }
    public Exception(String message, Throwable cause)  { super(message, cause); }
    public Exception(Throwable cause)                  { super(cause); }
}

3.1 受检异常 vs 非受检异常

java 复制代码
// 受检异常:编译器强制处理
public class CheckedExceptionDemo {
    public static void readFile() throws java.io.IOException {
        throw new java.io.IOException("文件不存在");
    }

    public static void main(String[] args) {
        try {
            readFile();
        } catch (java.io.IOException e) {
            System.err.println("I/O错误: " + e.getMessage());
        }
    }
}

// 非受检异常:编译器不强制
public class UncheckedExceptionDemo {
    public static void divide(int a, int b) {
        if (b == 0) {
            throw new IllegalArgumentException("除数不能为零");
        }
        System.out.println(a / b);
    }

    public static void main(String[] args) {
        divide(10, 2);   // OK
        // divide(10, 0); // 抛 IllegalArgumentException
    }
}

四、自定义异常最佳实践

java 复制代码
/**
 * 业务异常基类------统一异常体系
 */
public class BusinessException extends RuntimeException {
    private final String errorCode;
    private final transient Object[] args;

    public BusinessException(String errorCode, String message, Object... args) {
        super(message);
        this.errorCode = errorCode;
        this.args = args;
    }

    public String getErrorCode() { return errorCode; }
    public Object[] getArgs()    { return args; }

    public static BusinessException of(String errorCode, Object... args) {
        return new BusinessException(errorCode, format(errorCode, args), args);
    }

    private static String format(String code, Object... args) {
        return "[" + code + "] " + java.util.Arrays.toString(args);
    }

    public static void main(String[] args) {
        throw BusinessException.of("USER_001", "userId", 12345);
    }
}

五、面试要点

问题 关键要点
Error 是否应该捕获 通常不应捕获,OOM/StackOverflow 无法恢复
RuntimeException 和 Exception 区别 RuntimeException 是非受检异常,不强制处理
自定义异常选哪个父类 业务异常继承 RuntimeException,减少代码污染
相关推荐
IT_陈寒2 小时前
Redis内存爆了,原来我漏掉了这个致命配置
前端·人工智能·后端
fliter2 小时前
最后一块拼图:用 bitvec 构造 IPv4 包,真正做出自己的 Ping
后端
fliter3 小时前
用 Rust 解析并生成 ICMP 包:checksum、nom 与 cookie-factory
后端
蝎子莱莱爱打怪4 小时前
XZLL-IM干货系列 03|消息 ID 设计:一个 UUID 搞不定的事,我用两个 ID 解决了
后端·面试·开源
fliter4 小时前
从 panic 到 Result:用 Rust 重新整理一个 ping 项目的错误处理
后端
森蓝情丶4 小时前
我给 AI 搭了个法庭:一个前端仔的 LangGraph 实战全记录
前端·后端
JensCS猿4 小时前
从 Spring Boot 回看 SSM 框架:手动挡与自动挡的驾驶哲学
后端
爱勇宝4 小时前
干了近 8 年,一夜之间被裁:AI 时代,程序员最该害怕的不是 AI
前端·后端·程序员
科米米5 小时前
嵌入式日志模块
后端
血小溅5 小时前
三大 AI 编码框架深度对比:GSD vs OpenSpec vs Superpowers
人工智能·后端