异常:java.lang.NoClassDefFoundError: javax/xml/bind/DatatypeConverter

异常本质

jjwt-0.9.1 在生成签名时,需要调用 javax.xml.bind.DatatypeConverter.printBase64Binary(...) 做 Base64 编码。

JDK 9+ 开始 JAXB(java.xml.bind)被移出默认类路径,于是出现
java.lang.NoClassDefFoundError: javax/xml/bind/DatatypeConverter

一句话:老版本 jjwt 依赖了 JDK 8 自带、但 JDK 9+ 不再默认提供的 JAXB API。


最简修复方案(继续使用 jjwt-0.9.x)

pom.xml 把 JAXB 显式加回来:

xml 复制代码
<!-- 适用于 JDK 9/11/17/21 -->
<dependency>
    <groupId>javax.xml.bind</groupId>
    <artifactId>jaxb-api</artifactId>
    <version>2.3.1</version>
</dependency>
<!-- 如果还缺实现,再加一个实现包 -->
<dependency>
    <groupId>org.glassfish.jaxb</groupId>
    <artifactId>jaxb-runtime</artifactId>
    <version>2.3.9</version>
</dependency>

刷新 Maven → 重启 → DatatypeConverter 能找到,异常消失。


根治方案(推荐)

jjwt-0.9.1 已停止维护,官方 0.10+ 以后改用自己实现的 Base64,彻底去掉 JAXB 依赖

  1. 升级依赖(Spring Boot 3.x 示例,jjwt 0.12.5)
xml 复制代码
<dependency>
    <groupId>io.jsonwebtoken</groupId>
    <artifactId>jjwt-api</artifactId>
    <version>0.12.5</version>
</dependency>
<dependency>
    <groupId>io.jsonwebtoken</groupId>
    <artifactId>jjwt-impl</artifactId>
    <version>0.12.5</version>
    <scope>runtime</scope>
</dependency>
<dependency>
    <groupId>io.jsonwebtoken</groupId>
    <artifactId>jjwt-jackson</artifactId>
    <version>0.12.5</version>
    <scope>runtime</scope>
</dependency>
  1. 工具类微调(API 变化)
java 复制代码
// 生成
String token = Jwts.builder()
        .setClaims(claims)
        .signWith(SignatureAlgorithm.HS256, jwtProperties.getAdminSecretKey().getBytes(StandardCharsets.UTF_8))
        .setExpiration(new Date(System.currentTimeMillis() + jwtProperties.getAdminTtl()))
        .compact();

// 解析
Claims claims = Jwts.parserBuilder()
        .setSigningKey(jwtProperties.getAdminSecretKey().getBytes(StandardCharsets.UTF_8))
        .build()
        .parseClaimsJws(token)
        .getBody();

升级后无论 JDK 8 还是 JDK 21 都不再需要 JAXB。


快速判断选哪种

场景 做法
项目即将下线 / 懒得改代码 jaxb-api 依赖,5 分钟搞定
长期维护 / 想跑 JDK 17+ 直接升 jjwt 0.12.x(或 1.0 GA)

把依赖改完、重新打包,日志里就不会再出现 DatatypeConverterClassNotFoundException 了。

相关推荐
阿里巴巴P8高级架构师5 分钟前
从0到1:用 Spring Boot 4 + Java 21 打造一个智能AI面试官平台
java·后端
stevenzqzq7 分钟前
trace和Get thread dump的区别
java·android studio·断点
桦说编程7 分钟前
并发编程踩坑实录:这些原则,帮你少走80%的弯路
java·后端·性能优化
程序猿零零漆8 分钟前
Spring之旅 - 记录学习 Spring 框架的过程和经验(十三)SpringMVC快速入门、请求处理
java·学习·spring
BHXDML8 分钟前
JVM 深度理解 —— 程序的底层运行逻辑
java·开发语言·jvm
tkevinjd10 分钟前
net1(Java中的网络编程、TCP的三次握手与四次挥手)
java
码头整点薯条10 分钟前
基于Java实现的简易规则引擎(日常开发难点记录)
java·后端
J2虾虾18 分钟前
Java使用的可以使用的脚本执行引擎
java·开发语言·脚本执行
老马识途2.023 分钟前
java处理接口返回的json数据步骤 包括重试处理,异常抛出,日志打印,注意事项
java·开发语言
2***d88525 分钟前
Spring Boot中的404错误:原因、影响及处理策略
java·spring boot·后端