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 再次写入,从而重新触发启动卡死风险。
相关推荐
記億揺晃着的那天6 分钟前
Windows 通过 Java 获取可用端口的一个坑:Hyper-V 保留端口导致 UDP 绑定失败
java·windows·udp
组合缺一7 分钟前
SolonCode(编码智能体)支持鸿蒙 PC
java·华为·ai·ai编程·harmonyos·solon·soloncode
小bo波8 分钟前
用匿名内部类优雅地计算方法执行时间
java·设计模式·性能测试·模板方法模式·lambda·代码优化·匿名内部类
折哥的程序人生 · 物流技术专研13 分钟前
Tomcat 严重警告:JDBC 驱动未注销 + 工作线程泄漏 —— 原因、影响与彻底修复(生产级终极指南)
java·运维·数据库·mysql·oracle·tomcat
一个儒雅随和的男子16 分钟前
sentinel底层原理剖析以及实战优化
java·网络·sentinel
两年半的个人练习生^_^22 分钟前
JMM 进阶:彻底理解 synchronized 实现原理
java·开发语言
戳代码的新星29 分钟前
论小白如何学会使用Maven
java·maven
wyhwust29 分钟前
maven的安装和配置
java
plainGeekDev40 分钟前
HttpURLConnection → OkHttp + Kotlin
android·java·kotlin
swordbob40 分钟前
Spring Boot 2.0 改 CGLIB 后,接口实现是否有影响
java·开发语言·spring