jar启动卡死问题分析总结

问题

java -jar content.jarJDK21 环境下启动出现卡死/长时间无日志(CPU 高),同一份代码打出来的另一个 jar 不会卡死。

分析过程

  • 基础对比 :对比两份 jar 的 MANIFEST.MFMain-Class/Start-Class 一致;解压后文件结构一致。
  • 依赖一致性校验 :对比两边 BOOT-INF/lib148 个依赖 jar ,按 SHA256 逐文件校验 ,结论为完全一致(无缺失、无版本不一致、无同名不同内容)。
  • 定位卡死点 :抓取卡死进程线程栈,main 线程卡在 Spring Boot loader:
    • org.springframework.boot.loader.zip.ZipString.hash
    • ZipContent$Loader.loadContent
      说明卡死发生在 解析外层 jar 的 ZIP 目录/索引阶段,尚未进入业务启动逻辑。
  • 定位 jar 的关键差异 :解析两份 jar 的 ZIP entry extra fields,发现卡死 jar 大量包含:
    • extra header id=0x5455(ZIP 的 UT/Extended Timestamp
    • 996 个 entry 带 UT;不卡死 jar 不带 UT
  • 复现验证 :按构建方式(JDK21 + build 目录 mvn clean install)重新打包,产物再次出现 UT=996 ,启动复现卡死;调整打包输出时间戳后,验证可将 UT 降为 0,启动不再卡死并能正常输出日志。

说明:另一个 jar 启动时出现的 IllegalAccessError(模块强封装导致的 --add-opens 问题)属于不同问题,不作为本次"卡死"根因。

分析结论

  • 卡死原因不是依赖包/版本不一致 ,因为 BOOT-INF/lib 完全一致。
  • 卡死根因是:打包产物的 ZIP 元数据中写入了大量 UT(0x5455) 扩展时间戳 ,触发 Spring Boot loader 在解析 ZIP 内容时走到极慢路径,表现为卡在 ZipString.hash
  • UT 的出现与打包过程相关(归档条目时间戳/归档实现差异),而不是业务代码本身。

解决方案

  • 工程级(推荐) :在父 pom.xml 增加 Maven 属性,固定归档输出时间戳,避免 UT 写入并让产物稳定可复现:
xml 复制代码
<properties>
  <!-- 固定归档条目时间戳,避免 ZIP 扩展时间戳(UT/0x5455)写入 -->
  <project.build.outputTimestamp>1980-02-01T00:00:00Z</project.build.outputTimestamp>
</properties>
  • 验证结果 :重新 mvn clean install 后,产物 UT(0x5455)=0,启动不再卡死。
  • 可选:全局生效 :也可在 ~/.m2/settings.xml 的 profile 中注入同名属性并默认启用,让本机所有构建生效;但仍建议保留在项目 pom.xml,避免 CI/他人环境复现问题。示例:
xml 复制代码
<settings xmlns="http://maven.apache.org/SETTINGS/1.2.0"
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://maven.apache.org/SETTINGS/1.2.0 https://maven.apache.org/xsd/settings-1.2.0.xsd">
  <profiles>
    <profile>
      <id>reproducible-jar</id>
      <properties>
        <project.build.outputTimestamp>1980-02-01T00:00:00Z</project.build.outputTimestamp>
      </properties>
    </profile>
  </profiles>

  <!-- 默认启用(也可以不写这里,改为构建时加 -Preproducible-jar) -->
  <activeProfiles>
    <activeProfile>reproducible-jar</activeProfile>
  </activeProfiles>
</settings>
  • 可选:命令行动态覆盖(不推荐用于解决卡死) :只对本次构建 生效,可以用 -D 覆盖 project.build.outputTimestamp
bash 复制代码
mvn clean install -Dproject.build.outputTimestamp="$(date -u +%Y-%m-%dT%H:%M:%SZ)"

说明:

  • 这会让每次构建的"归档条目时间戳"随当前时间变化。
  • 由于本问题与 ZIP 的 UT(0x5455) 扩展时间戳强相关,动态时间戳可能导致 UT 再次写入,从而重新触发启动卡死风险。
相关推荐
小北方城市网3 小时前
Redis 分布式锁高可用实现:从原理到生产级落地
java·前端·javascript·spring boot·redis·分布式·wpf
六义义4 小时前
java基础十二
java·数据结构·算法
毕设源码-钟学长5 小时前
【开题答辩全过程】以 基于SpringBoot的智能书城推荐系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
笨手笨脚の5 小时前
深入理解 Java 虚拟机-03 垃圾收集
java·jvm·垃圾回收·标记清除·标记复制·标记整理
莫问前路漫漫5 小时前
WinMerge v2.16.41 中文绿色版深度解析:文件对比与合并的全能工具
java·开发语言·python·jdk·ai编程
九皇叔叔6 小时前
【03】SpringBoot3 MybatisPlus BaseMapper 源码分析
java·开发语言·mybatis·mybatis plus
挖矿大亨6 小时前
c++中的函数模版
java·c++·算法
a程序小傲6 小时前
得物Java面试被问:RocketMQ的消息轨迹追踪实现
java·linux·spring·面试·职场和发展·rocketmq·java-rocketmq
青春男大6 小时前
Redis和RedisTemplate快速上手
java·数据库·redis·后端·spring·缓存
Ghost Face...6 小时前
i386 CPU页式存储管理深度解析
java·linux·服务器