在Java里如何选择合适的JDK版本_Java版本选择与区别说明

JDK版本选择应基于项目生命周期、依赖支持和运行环境约束,当前生产推荐JDK 17(LTS),新项目可选JDK 21(LTS);需重点验证依赖兼容性、构建配置与实际运行时JDK一致性。

选 JDK 版本不是看最新,而是看项目生命周期、依赖库支持和运行环境约束。JDK 17(LTS)是当前最稳妥的生产选择,JDK 21(LTS)适合新项目且需虚拟线程等特性,JDK 8 已不建议新增项目使用。

怎么看项目依赖是否支持某个JDK

关键不是“能不能跑”,而是“会不会出隐性问题”。很多老库在 JDK 11+ 上会因模块系统(java.base 默认强封装)、移除 API(如 javax.xml.bind)或反射限制(--illegal-access=deny)而失败。

  • 检查构建日志里是否有 WARNING: Using incubator modulesClassNotFoundException: javax.xml.bind.JAXBContext
  • Maven 项目运行 mvn dependency:tree,再搜索是否含已弃用的依赖,例如 com.sun.xml.bind:jaxb-impl(JDK 11+ 需显式引入)
  • java -XX:+ShowHiddenFrames -version 启动应用,观察是否触发 InaccessibleObjectException

JDK 8 / 11 / 17 / 21 的核心差异点

版本跳变带来的是兼容性断层,不是功能叠加。比如从 JDK 8 升到 11,java.awt.HeadlessException 可能在无 GUI 环境下突然抛出;从 17 到 21,Virt

ualThread 默认调度策略变更可能影响 ExecutorService 行为。

  • JDK 8:默认使用 Parallel GC,无模块系统,String 内部是 char[]
  • JDK 11:首次 LTS 含 HTTP Client (java.net.http),移除 Java EECorba 模块,JVMCI 接口开放
  • JDK 17:密封类(sealed)、强封装 internal.* 包、Switch 表达式正式化、G1 成为默认 GC
  • JDK 21:虚拟线程(VirtualThread)正式可用、ScopedValue 替代 InheritableThreadLocalString Templates(预览转正式)

如何验证本地 JDK 是否真被项目使用

IDE 或构建工具常缓存旧配置,导致你以为用了 JDK 21,实际编译/运行仍是 JDK 11。

  • Maven:检查 pom.xmlmaven-compiler-plugin,它们只控制字节码版本,不决定运行时 JDK
  • Gradle:确认 java.toolchain.version = "21",而非仅 sourceCompatibility = JavaVersion.VERSION_21
  • 运行时验证:在启动类加 System.out.println(System.getProperty("java.version"));,或执行 ps aux | grep java 查看进程实际加载的 java 路径
public class JdkCheck {
    public static void main(String[] args) {
        System.out.println("JDK version: " + System.getProperty("java.version"));
        System.out.println("Java home: " + System.getProperty("java.home"));
        System.out.println("Runtime name: " + System.getProperty("java.runtime.name"));
    }
}

真正容易被忽略的是 CI/CD 环境中的 JDK 版本——它往往和本地开发不一致,而且错误日志里不会直接写“你用了错的 JDK”,只会报 NoClassDefFoundErrorUnsupportedClassVersionError,得倒推排查。