Java异常的抛出与捕获机制

throw是抛出异常对象的语句,位于方法体内;throws是声明异常类型的关键字,位于方法签名末尾;前者用于触发异常,后者用于告知调用方可能抛出的受检异常。

throw 和 throws 的区别到底在哪

很多人写到一半发现编译报错,加了 throw new Exception() 却被 IDE 提示“Unhandled exception”,其实是混淆了 throw(抛出一个异常对象)和 throws(声明方法可能抛出的异常类型)。

  • throw 是语句,只能出现在方法体内部,后面跟的是 Exception 实例,比如 throw new IllegalArgumentException("id 不能为负")
  • throws 是方法签名的一部分,写在方法声明末尾,后面跟的是异常类名,可以多个,用逗号分隔,比如 public void readFile() throws IOException, SecurityException
  • 只有受检异常(Exception 及其子类,但不包括 RuntimeException)强制要求处理:要么用 try-catch 捕获,要么用 throws 声明上抛
  • RuntimeException 及其子类(如 NullPointerExceptionArrayIndexOutOfBoundsException)属于非受检异常,编译器不强制处理,但运行时仍会中断流程

try-catch-finally 中哪些部分可以省略

catchfinally 都不能单独和 try 搭配;必须至少有一个。常见组合只有两种合法形式:try-catchtry-catch-finally(或 try-finally)。

  • try-finally 合法,常用于确保资源释放,比如关闭 FileInputStream,即使前面发生异常也要执行 finally
  • try-catch 合法,用于拦截并处理异常,但要注意:如果 catch 里没重新抛出,异常就终止了,上层看不到
  • try 块中 return 语句遇到 finally 时,会先执行 finally 再返回——但如果 finally 里也有 return,它会覆盖 trycatch 中的返回值
  • JDK 7+ 支持 try-with-resources,自动关闭实现了 AutoCloseable 的资源,比手写 finally 更安全,推荐优先使用

捕获异常时为什么不能只写 catch(Exception e)

看似“一网打尽”,实际会掩盖真正的问题,让调试变困难,也容易误吞本该向上抛的受检异常。

  • 过度宽泛的 catch(Exception e) 会捕获 RuntimeException 和所有受检异常,包括本不该在此处处理的 IOExceptionSQLException
  • 如果业务逻辑明确知道只可能出 NumberFormatException,就该只捕获它,而不是用 Exception 包一层再忽略
  • 多异常捕获从 JDK 7 开始支持语法糖:catch (IOException | SQLException e),比分别写两个 catch 更简洁,但前提是它们处理方式一致
  • 日志记录时建议用 e.printStackTrace() 仅用于调试;生产环境应使用 logger.error("读取配置失败", e),保留完整堆栈

自定义异常该继承 RuntimeException 还是 Exception

取决于你是否希望调用方**必须**处理它。这是设计契约,不是技术限制。

立即学习“Java免费学习笔记(深入)”;

  • 继承 RuntimeException → 非受检异常 → 调用方可选择处理或不处理,适合程序逻辑错误(如参数校验失败、状态非法),例如 InvalidOrderStatusException
  • 继承 Exception → 受检异常 → 编译器强制调用方处理,适合外部不确定性问题(如网络超时、文件不存在),例如 PaymentTimeoutException(如果业务要求每个支付调用都显式应对超时)
  • 无论哪种,都建议提供至少两个构造函数:一个带 String message,一个带 String messageThrowable cause,方便链式异常追踪
  • 不要为了“看起来规范”而滥用受检异常——如果大多数调

    用方只会写 catch(Exception e) { throw new RuntimeException(e); },那不如一开始定义成非受检的
public class InsufficientBalanceException extends RuntimeException {
    public InsufficientBalanceException(String message) {
        super(message);
    }
    public InsufficientBalanceException(String message, Throwable cause) {
        super(message, cause);
    }
}

异常机制不是用来代替 if 判断的,也不是为了让代码“看起来健壮”。关键在两点:该由谁处理、能否真正恢复。很多空 catch 块和泛化捕获,反而让故障更难定位。