Java中抛出异常的核心是:用throw抛出异常实例,用throws声明受检异常;未捕获的受检异常必须声明,否则编译失败;运行时异常无需声明。

Java 方法中抛出异常,核心就两点:用 throw 主动抛出具体异常实例,或用 throws 声明可能向上级传递的受检异常。不声明、不处理受检异常(Exception 及其子类,但不包括 RuntimeException)会导致编译失败。
什么时候必须写 throws?
当方法体内调用了会抛出受检异常(checked exception)的代码,且你没用 try-catch 捕获它,就必须在方法签名后加 throws 声明。否则编译器直接报错。
-
IOException、SQLException、ClassNotFoundException都属于受检异常 -
NullPointerException、ArrayIndexOutOfBoundsException是运行时异常(unchecked),无需throws也能编译通过 - 一个方法可声明多个异常,用逗号分隔:
throws IOException, SQLException
throw 和 throws 的区别与配合
throw 是语句,用于抛出一个异常对象;throws 是修饰符,用于方法声明。二者常一起出现:你在方法里 throw 一个受检异常,就得在方法上 throws 它。
public void readFile(String path) throws IOException {
if (path == null) {
throw new IllegalArgumentException("path cannot be null");
}
FileInputStream fis = new FileInputStream(path); // 可能抛出 IOException
}
-
throw后面跟的是异常实例(必须是Throwable或其子类) -
throws后面跟的是异常类型(类名),不是实例 - 可以
throw运行时异常而不写throws;但若throw受检异常,又没捕获,就必须throws
自定义异常怎么抛?
继承 Exception 就是受检异常,必须声明或捕获;继承 RuntimeException 就是运行时异常,可选处理。
立即学习“Java免费学习笔记(深入)”;
class InvalidAgeException extends Exception {
public InvalidAgeException(String msg) {
super(msg);
}
}
class TooYoungException extends RuntimeException {
public TooYoungException(String msg) {
super(msg);
}
}
public void checkAge(int age) throws InvalidAgeException {
if (age < 0) {
throw new InvalidAgeException("Age cannot be negative");
}
if (age < 18) {
throw new TooYoungException("Must be 18 or older");
}
}
- 自定义受检异常(如
InvalidAgeException)强制调用方处理,适合业务规则硬性约束 - 自定义运行时异常(如
TooYoungException)更轻量,适合开发阶段快速反馈,或调用方本就不该恢复的场景 - 不要为了“看起来规范”而把所有异常都做成受检的——过度使用
throws会让调用代码充斥try-catch或层层往上推,反而掩盖真正需要处理的问题
最容易被忽略的是:受检异常的传播路径一旦中断(比如某层用 catch 吞掉却没重抛或记录),错误信息就丢失了;而运行时异常若没被任何地方捕获,会一路冒泡到线程终止,日志里只有一行堆栈——所以抛什么异常、在哪抛、是否包装原始异常(用 new XxxException("msg", cause)),比单纯语法更重要。










