throw用于方法体内主动抛出异常对象,后接Throwable实例;检查型异常需throws声明或try-catch处理,运行时异常可直接抛出。
当你需要在代码执行过程中手动触发一个异常,比如校验失败、状态非法时,就用 throw。它后面必须跟一个 Throwable 或其子类的实例,不能是类名或字符串。
常见错误现象:throw new Exception(); 编译不通过(未处理检查型异常);throw "error"; 直接报错(类型不匹配)。
throw 只能抛出一个异常对象,不能抛出多个IOException),当前方法必须用 throws 声明,或用 try-catch 捕获IllegalArgumentException)可直接 throw,无需声明public void setAge(int age) {
if (age < 0) {
throw new IllegalArgumentException("年龄不能为负数");
}
this.age = age;
}
throws 不抛异常,只是“提前报备”——告诉调用者:“我这个方法可能会甩出这些异常,你得负责处理”。它只对检查型异常(Exception 及其子类,但不包括 RuntimeException)有强制约束力。
使用场景:文件读写、网络请求、反射调用等底层操作,通常由 JDK API 抛出 IOException、SQLException 等。
throws 多个异常,用逗号分隔:throws IOException, SQLException
throws 不影响运行时行为,只是编译期契约;运行时是否真抛,取决于实际执行路径throws 的异常类型不能比父类更宽泛(只能相同或更具体)public String readFile(String path) throws IOException {
return Files.readString(Paths.get(path));
}
绝大多数真实场景里,二者配合使用:方法内部用 throw 触发异常,方法签名用 throws 向外声明。这不是设计缺陷,而是 Java 强制检查型异常处理机制的体现。
容易踩的坑:throws Exception 过度宽泛,掩盖真实异常类型;或者在已经 catch 住异常后,又无意义地 throw 出去却不记录日志。
FileNotFoundException 而非笼统的 IOException
catch 块中重新抛出,建议用 throw new XxxException("msg", e) 保留原始栈轨迹throws Exception,这会让调用方无法针对性处理像 NullPointerException、ArrayIndexOutOfBoundsException、IllegalArgumentException 这类,继承自 RuntimeException,Java 不强制要求声明或捕获。

throw,调用方也完全可以选择忽略。
但这不等于“应该忽略”——过度依赖运行时异常会削弱接口契约,增加排查成本。
IllegalStateException),避免污染方法签名Exception 还是 RuntimeException
throws IOException 盲目一路向上扔,却没在关键节点做语义转换或上下文补充。