throw语句只能抛出Throwable或其子类的实例,如new IllegalArgumentException("msg"),不可抛出普通对象或null,否则编译失败或运行时抛NullPointerException;检查型异常需配合throws声明,运行时异常可不声明但建议注明。

throw 语句只能抛出 Throwable 或其子类的实例
Java 中 throw 不是“扔任意对象”,它后面必须跟一个已实例化的异常对象。常见错误是直接写 throw IllegalArgumentException;(没加 new)或传入 null,这会导致编译失败或运行时 NullPointerException。
-
throw后面必须是Throwable实例,比如new IllegalArgumentException("msg")、new IOException() - 不能 throw 普通类(如
throw new String("oops")),编译不通过 - 若构造函数带参数,注意检查签名:比如
IllegalArgumentException接受String或String + Throwable,但不接受纯int
throw 和 throws 的分工要分清
throw 是“抛出一个异常实例”,throws 是“声明方法可能甩出哪些异常类型”。二者常一起用,但作用完全不同。混淆会导致编译报错或掩盖真正问题。
- 方法体内用
throw;方法签名末尾用throws声明检查型异常(如IOException) - 如果
throw的是检查型异常(Exception及其子类,但非RuntimeException),必须在方法上加throws,否则编译失败 -
RuntimeException及其子类(如NullPointerException、IllegalArgumentException)可不声明throws,但写上更利于调用方理解
主动 throw 的典型场景和写法
不是所有异常都该手动 throw,重点在于“提前拦截非法状态”,避免后续逻辑崩溃或产生歧义结果。
public void setAge(int age) {
if (age < 0 || age > 150) {
throw new IllegalArgumentException("age must be between 0 and 150, got: " + age);
}
this.age = age;
}
- 参数校验失败时 throw
IllegalArgumentException(运行时异常,无需throws) - 资源不可用时 throw
IllegalStateException(例如调用close()后再操作) - 业务规则不满足时,可自定义异常并 throw,比如
throw new InsufficientBalanceException("balance too low") - 避免在循环里频繁 throw——性能差,且掩盖真实上下文;应先收集问题,再统一抛出
throw null 会触发 NullPointerException
这是容易被忽略的陷阱:虽然语法上 throw 后接表达式,但如果该表达式求值为 null,JVM 会在运行时抛出 NullPointerException,而不是你期望的那个异常。
立即学习“Java免费学习笔记(深入)”;
Exception e = null; throw e; // 编译通过,但运行时报 NullPointerException,不是 e 本身
- 永远不要让
throw后的表达式可能为null - 如果异常对象来自变量或方法返回值,务必判空或确保非空(如用 Optional 或断言)
- IDE 通常不会警告这个风险,得靠代码审查或单元测试暴露
throw 时,最常踩的坑不是语法,而是没想清楚“这个异常是否真该由当前层抛出”——有时该记录日志后静默处理,有时该包装成更高层语义的异常再 throw。










