JDK 从 8 升级到 21 的问题集
一、背景与挑战
二、进度
应用总数 | 已完成 | 应用下线 | 待升级 |
100+ | 73 | 13 | 10+ |
三、主要问题域与解决方案
1. 依赖管理的"蝴蝶效应"
核心原因在于JEP320提案:https://openjdk.org/jeps/320
案例1:历史SDK的编译陷阱
Compilation failure: Compilation failure:
#14 4.173 [ERROR] 不再支持源选项 6。请使用 8 或更高版本。
#14 4.173 [ERROR] 不再支持目标选项 6。请使用 8 或更高版本。
<!-- 旧版本编译器配置导致构建失败 -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.5</version>
<configuration>
<source>1.6</source>
<target>1.6</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.13.0</version>
<configuration>
<release>8</release><!-- 统一使用release参数 -->
</configuration>
</plugin>
运行 HTML
案例2:JAXB的模块化剥离
javax.xml.bind.JAXBException:Implementation of JAXB-API has not been found
<dependency>
<groupId>org.glassfish.jaxb</groupId>
<artifactId>jaxb-runtime</artifactId>
<version>4.0.5</version>
</dependency>
案例3:Lombok与新版编译器兼容性问题
java: java.lang.NoSuchFieldError
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.30</version>
</dependency>
案例4:Resource注解找不到
Caused by: java.lang.NoSuchMethodError: 'java.lang.String javax.annotation.Resource.lookup()'
at org.springframework.context.annotation.CommonAnnotationBeanPostProcessor$ResourceElement.<init>(CommonAnnotationBeanPostProcessor.java:664)
at org.springframework.context.annotation.CommonAnnotationBeanPostProcessor.lambda$buildResourceMetadata$0(CommonAnnotationBeanPostProcessor.java:395)
at org.springframework.util.ReflectionUtils.doWithLocalFields(ReflectionUtils.java:669)
at org.springframework.context.annotation.CommonAnnotationBeanPostProcessor.buildResourceMetadata(CommonAnnotationBeanPostProcessor.java:377)
at org.springframework.context.annotation.CommonAnnotationBeanPostProcessor.findResourceMetadata(CommonAnnotationBeanPostProcessor.java:358)
at org.springframework.context.annotation.CommonAnnotationBeanPostProcessor.postProcessMergedBeanDefinition(CommonAnnotationBeanPostProcessor.java:306)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.applyMergedBeanDefinitionPostProcessors(AbstractAutowireCapableBeanFactory.java:1116)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:594)
... 37 more
<dependency>
<groupId>jakarta.annotation</groupId>
<artifactId>jakarta.annotation-api</artifactId>
<version>1.3.5</version>
</dependency>
<dependency>
<groupId>javax.annotation</groupId>
<artifactId>javax.annotation-api</artifactId>
<version>1.3.2</version>
</dependency>
上述两个依赖代码基本一样,推荐使用该版本:
jakarta.annotation:jakarta.annotation-api。
2. 模块化的破与立
反射访问的模块墙
[ERROR] Unable to make field private int java.text.SimpleDateFormat.serialVersionOnStream accessible
# 启动参数添加模块开放配置
--add-opens java.base/java.text=ALL-UNNAMED
--add-opens java.base/java.lang.reflect=ALL-UNNAMED
完整模块开放配置模板
export JAVA_OPTS="-Djava.library.path=/usr/local/lib -server -Xmx4096m --add-opens java.base/sun.security.action=ALL-UNNAMED
--add-opens java.base/java.lang=ALL-UNNAMED
--add-opens java.base/java.math=ALL-UNNAMED
--add-opens java.base/java.util=ALL-UNNAMED
--add-opens java.base/sun.util.calendar=ALL-UNNAMED
--add-opens java.base/java.util.concurrent=ALL-UNNAMED
--add-opens java.base/java.util.concurrent.locks=ALL-UNNAMED
--add-opens java.base/java.security=ALL-UNNAMED
--add-opens java.base/jdk.internal.loader=ALL-UNNAMED
--add-opens java.management/com.sun.jmx.mbeanserver=ALL-UNNAMED
--add-opens java.base/java.net=ALL-UNNAMED
--add-opens java.base/sun.nio.ch=ALL-UNNAMED
--add-opens java.management/java.lang.management=ALL-UNNAMED
--add-opens jdk.management/com.sun.management.internal=ALL-UNNAMED
--add-opens java.management/sun.management=ALL-UNNAMED
--add-opens java.base/sun.security.action=ALL-UNNAMED
--add-opens java.base/sun.net.util=ALL-UNNAMED
--add-opens java.base/java.time=ALL-UNNAMED
--add-opens java.base/java.lang.reflect=ALL-UNNAMED
--add-opens java.base/java.io=ALL-UNNAMED"
3. 语法层面的"时空穿越"
案例1:Base64编解码改造
// JDK8写法(已废弃)
BASE64Encoder encoder =newBASE64Encoder();
String encoded = encoder.encode(data);
// JDK21规范写法
Base64.Encoder encoder =Base64.getEncoder();
String encoded = encoder.encodeToString(data);
案例2:日期序列化问题
Caused by:java.lang.reflect.InaccessibleObjectException:
Unable to make field private int java.text.SimpleDateFormat.serialVersionOnStream accessible
解决方案
--add-opens java.base/java.text=ALL-UNNAMED
4. 隐秘的"依赖战争"
注解包冲突典型案例
[ERROR] javax.annotation.Resource exists in both
jsr250-api-1.0.jar and jakarta.annotation-api-1.3.5.jar
<!-- 统一使用Jakarta标准 -->
<dependency>
<groupId>jakarta.annotation</groupId>
<artifactId>jakarta.annotation-api</artifactId>
<version>2.1.1</version>
</dependency>
<!-- 排除旧版本依赖 -->
<exclusions>
<exclusion>
<groupId>javax.annotation</groupId>
<artifactId>jsr250-api</artifactId>
</exclusion>
</exclusions>
5. 构建体系的改造
Maven插件兼容性问题
[ERROR] The plugin org.apache.maven.plugins:maven-compiler-plugin:3.13.0
requires Maven version 3.6.3
升级策略
<build>
<pluginManagement>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.13.0</version>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<version>3.4.0</version>
</plugin>
</plugins>
</pluginManagement>
</build>
四、最佳实践总结
1. 本地编译
第一步:在本地进行编译,提前识别出语法错误、版本冲突及不兼容问题。
主要有以下几种场景:
Base64:参照 【Base64编解码改造】
lombok:升级版本
jsr250、jaxb-runtime、jakarta.annotation-api:参照 【注解包冲突典型案例】
maven-compiler-plugin:升级版本
maven-resources-plugin:升级版本
maven-war-plugin:升级版本
2. 行云构建
同【本地编译】
3. 行云部署
a、镜像不匹配:自定义镜像或者使用已申请的jdk21镜像
b、module权限不够:参照【完整模块开放配置模板】
c、JDSecurity加解密
所有数据库操作:important.properties配置文件的处理方式
classpath:important.properties 使用PropertyPlaceholderConfigurer进行处理,不要用JDSecurityPropertyFactoryBean。
<!-- <bean id ="secApplicationProperties" class="com.jd.security.configsec.spring.config.JDSecurityPropertyFactoryBean">-->
<!-- <property name="ignoreResourceNotFound" value="true" />-->
<!-- <property name="secLocation" value="classpath:important.properties"/>-->
<!-- </bean>-->
4. 运行
a、序列化异常
jdk21使用列表视图作为入参,导致jsf接口进行反序列化报错。报错代码如下:
List<String> subList = venderCodes.subList(i * batchSize, Math.min(venderCodes.size(), (i + 1) * batchSize));
VendorQueryVo vendorQueryVo = new VendorQueryVo();
vendorQueryVo.setVendorCodes(subList);
// 该接口最多支持100条调用
List<VendorVo> batchVendorNameByVendorCode = vendorBaseInfoService.getBatchVendorNameByVendorCode(vendorQueryVo, I18NParamFactory.getJDI18nParam());
将 vendorQueryVo.setVendorCodes(subList) 修改为vendorQueryVo.setVendorCodes(new ArrayList<>(subList)) 即可解决问题
b、线程上下文类找不到:使用多线程场景下尽可能使用显式指定线程池【默认情况下 不同运行环境的处理机制不同】
5. JVM调优
垃圾回收调优
UseParallelGC
、UseG1GC
和UseZGC
是 Java 虚拟机(JVM)中三种不同的垃圾回收器(Garbage Collector, GC),它们的设计目标和使用场景有所不同。以下是它们的区别:
特性 | UseParallelGC | UseG1GC | UseZGC |
设计目标 | 高吞吐量 | 平衡吞吐量和延迟 | 极低延迟 |
暂停时间 | 较长 | 较短 | 极短 |
适用堆大小 | 中小堆(几 GB 到几十 GB) | 大堆(几十 GB 到几百 GB) | 超大堆(TB 级别) |
CPU 消耗 | 中等 | 中等 | 较高 |
适用场景 | 批处理、计算密集型任务 | 对延迟有一定要求的应用 | 对延迟极其敏感的应用 |
仅供参考,具体请按照实际情况来进行调整。
6. RSS内存异常
RSS内存有问题的应用可以尝试使用jemalloc方式替换linux默认的glibc内存分配方式。
问题现象:启动后RSS内存使用率很快达到90%以上,甚至接近于100%。
使用方式:启动脚本增加
if [ ! -f "/usr/local/lib/libjemalloc.so.2" ]; then sudo wget -O /usr/local/lib/libjemalloc.so.2 http://storage.jd.local/ste/jemalloc/x86/5.3.0/libjemalloc.so.2 fi export LD_PRELOAD=$LD_PRELOAD:/usr/local/lib/libjemalloc.so.2
使用效果:
默认模式下稳定后的RSS内存在80%左右,替换后稳定在50%左右,且容器内存使用率也有所变化。
https://taishan.jd.com/mdc/ipMonitor?ip=11.31.93.15&startTime=1747238400000&endTime=1747361529820

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
-
上一篇
马斯克旗下 xAI 公司挖角英伟达研发世界模型
马斯克旗下xAI公司被曝从英伟达挖角两名核心研究员,加速研发“世界模型”,目标应用于AI生成游戏及机器人领域。 据报道,xAI从英伟达挖走研究员Zeeshan Patel和Ethan He。两人曾参与英伟达Omniverse平台开发,该平台以物理仿真技术著称,为世界模型研发提供关键技术基础。 Zeeshan Patel专攻多模态模型与物理AI,Ethan He的研究方向涵盖视频自监督学习与多模态模型,两人均具备世界模型开发经验。 据了解,xAI计划通过世界模型实现“物理规律理解”,使AI能模拟真实环境并实时交互。首批应用聚焦游戏领域,目标2026年底推出由AI动态生成的3D游戏,场景可随玩家行为实时变化。长期规划中,该技术或延伸至机器人、自动驾驶等领域,与特斯拉、Neuralink等马斯克旗下企业形成协同。 世界模型被视为实现通用人工智能(AGI)的关键路径,谷歌、Meta、英伟达等巨头已布局。英伟达2025年1月发布的Cosmos平台训练数据达9000万亿token,覆盖机器人、驾驶等场景。
-
下一篇
通义千问、豆包内测记忆功能,全面对标 ChatGPT
据 01Founder 获悉,国内两家科技巨头——阿里巴巴和字节跳动旗下的 AI 助手通义千问(Qwen)和豆包(Doubao),同时开始内测“记忆功能”。 此举被广泛视为对标行业领头羊 OpenAI 的 ChatGPT,标志着国产 AI 助手正从“即时问答工具”向“长期私人助理”的角色加速演进。据悉,OpenAI 在今年 4 月为 ChatGPT 更新了记忆功能。 报道指出,Qwen 和豆包几乎在同一时间点亮“记忆”技能树,绝非巧合,而是国内大模型应用进入下半场,从追求模型参数转向打磨用户体验的必然结果。 据一位参与灰度测试的用户透露,Qwen 的记忆功能已在小范围内悄然上线。从内测截图显示,其交互逻辑直观且主动。报道分析,这表明 Qwen 采取了“显式记忆”的策略,即由用户主动触发,AI 进行确认和存储。 另外,有俄语 App 用户也收到了该内测功能,对此报道表示,这一细节暗示,Qwen 的记忆功能测试可能并非仅限中文环境。 豆包方面,其官网在 10 月 10 日同步更新了《隐私政策》并上线了《记忆功能 FAQ》页面。在这些文件中,豆包向用户详细阐明了记忆的范围(不含图片、文件等)...
相关文章
文章评论
共有0条评论来说两句吧...