Java 17+ JDK 不再提供独立 JRE,java 命令直接来自 JDK/bin;JAVA_HOME 必须指向 JDK 根目录而非 bin,旧版 JRE 路径、rt.jar/tools.jar 引用及未适配模块系统的反射调用将导致运行或编译失败。
JDK 安装不再提供 JRE,java 命令直接来自 JDK 自带运行时从 JDK 14 开始,Oracle 就移除了独立 JRE 下载;到 JDK 17(LTS)及后续版本(如 21、22),JDK 安装包默认只含 bin、lib、conf 等目录,没有 jre 子目录。这意味着:
java、javac、jar 全部位于 JDK_HOME/bin,无需额外配置 JRE_HOME
JRE_HOME 环境变量(比如某些 IDE 启动脚本或 CI 配置),会报 Command not found 或路径不存在错误JRE/bin/java 启动,会因模块系统缺失而抛 java.lang.NoClassDefFoundError: java/sql
/SQLException 类似错误JAVA_HOME 必须指向 JDK 根目录,不是 bin 目录常见误操作:把 JAVA_HOME 设为 /path/to/jdk-17.0.1/bin,导致 javac 找不到 tools.jar(已废弃)或模块描述符,实际编译失败。
JAVA_HOME=/Library/Java/JavaVirtualMachines/jdk-17.0.1.jdk/Contents/Home(macOS)或 C:\Program Files\Java\jdk-17.0.1(Windows)echo $JAVA_HOME && $JAVA_HOME/bin/java -version,输出应为 JDK 版本而非报错
Project Structure → SDKs 里重新添加 JDK 路径,不能复用旧的 JRE 配置java.base 等)启用后,--add-modules 和 --add-opens 成为高频参数JDK 9 引入模块系统,JDK 17+ 默认严格限制反射与内部 API 访问。很多老框架(如 Hibernate 5.4、Log4j 2.16 之前)会因 java.lang.reflect.InaccessibleObjectException 崩溃。
--add-opens java.base/java.lang=ALL-UNNAMED --add-opens java.base/java.util=ALL-UNNAMED
--add-modules java.se.ee(仅限 JDK 17,JDK 21 已移除该聚合模块)build.gradle 中显式配置:java { toolchain { languageVersion = JavaLanguageVersion.of(17) } },否则 javac 可能默认用 1.8 源码级别rt.jar 和 tools.jar 彻底消失,依赖它们的构建脚本会直接失败如果你的 CI 脚本或 Ant 配置里写了 rt.jar 路径(如 ${JAVA_HOME}/jre/lib/rt.jar)或显式引用 tools.jar,JDK 17+ 运行时会报 FileNotFoundException。
java.base 等命名模块提供,可通过 java --list-modules 查看sun.misc.Launcher.getBootstrapClassPath(),需改用 ModuleLayer.boot().modules() 获取启动模块maven-compiler-plugin 若未声明 17 和 17 ,编译产物可能不兼容新 JVM