在Java里如何主动抛出异常_Javathrow关键字用法说明

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 及其子类(如 NullPointerExceptionIllegalArgumentException)可不声明 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,而不是你期望的那个异常。

Exception e = null;
throw e; // 编译通过,但运行时报 NullPointerException,不是 e 本身
  • 永远不要让 throw 后的表达式可能为 null
  • 如果异常对象来自变量或方法返回值,务必判空或确保非空(如用 Optional 或断言)
  • IDE 通常不会警告这个风险,得靠代码审查或单元测试暴露
实际写 throw 时,最常踩的坑不是语法,而是没想清楚“这个异常是否真该由当前层抛出”——有时该记录日志后静默处理,有时该包装成更高层语义的异常再 throw。