0

0

Java如何生成和验证XML签名

幻夢星雲

幻夢星雲

发布时间:2026-01-16 11:15:50

|

547人浏览过

|

来源于php中文网

原创

最稳妥的是jdk内置javax.xml.crypto.dsig包,需用xmlsignaturefactory等类,私钥须为privatekey类型,canonicalization方法须显式指定,uri为空表示全文档签名,签名前需normalize()。

java如何生成和验证xml签名

Java生成XML签名用什么API最稳妥

直接用 javax.xml.crypto.dsig 包,这是JDK内置的W3C XML Signature标准实现,无需额外依赖,兼容Java 8+。别用第三方XML库自己拼接签名节点——签名值、引用摘要、Canonicalization算法都必须严格遵循规范,手写极易出错。

关键类有:XMLSignatureFactoryDOMSignContextReferenceSignedInfoKeyInfo。私钥必须是 PrivateKey 类型(如 PKCS8EncodedKeySpec 解析后的 RSA PrivateKey),不能传入字符串或PEM文本。

  • 必须显式指定 Canonicalization 方法,常用 CanonicalizationMethod.INCLUSIVE;不设默认值可能在不同JDK版本行为不一致
  • ReferenceURI 属性为空字符串("")表示对整个文档签名;若指向子元素(如 "#order"),目标元素必须含 Id 属性且已设 setIdAttribute("Id", true)
  • 签名前需调用 document.getDocumentElement().normalize(),否则某些节点顺序/空白处理会导致验证失败

Java验证XML签名时常见失败原因

验证失败不是“签名错”,大概率是上下文不匹配。核心检查点只有三个:密钥、规范化方式、引用URI解析路径。

  • 公钥必须和签名时私钥配对,且类型匹配(如签名用RSA-SHA256,验证时 KeyInfo 中的 X509Data 必须能正确加载对应公钥)
  • 验证用的 DOMValidateContext 必须传入原始XML文档的 Document 对象,不能是字符串再parse一次——DOM树对象引用丢失会导致 Id 属性无法定位
  • 如果签名中 ReferenceURI 是相对路径(如 "data.xml"),验证时需通过 setBaseURI() 显式设置基础路径,否则 resolveResource 找不到外部资源
  • 错误信息如 "the signature is invalid""reference has no corresponding element" 都指向上述某一项配置偏差,不是算法问题

如何用Java签名并嵌入到现有XML文档

签名不是追加一段字符串,而是构造标准 <signature></signature> 元素并插入到指定位置(通常作为子元素或同级元素)。插入点由 DOMSignContext 的构造参数决定。

IBM Watson
IBM Watson

IBM Watson文字转语音

下载

立即学习Java免费学习笔记(深入)”;

Document doc = DocumentBuilderFactory.newInstance().newDocumentBuilder().parse(new File("input.xml"));
Element root = doc.getDocumentElement();
root.setAttribute("Id", "root"); // 若准备签整个文档,设Id便于引用

XMLSignatureFactory fac = XMLSignatureFactory.getInstance("DOM");
Reference ref = fac.newReference("", fac.newDigestMethod(DigestMethod.SHA256, null),
    Collections.singletonList(fac.newTransform(Transform.ENVELOPED, (TransformParameterSpec) null)),
    null, null);

SignedInfo si = fac.newSignedInfo(
    fac.newCanonicalizationMethod(CanonicalizationMethod.INCLUSIVE, (C14NMethodParameterSpec) null),
    fac.newSignatureMethod(SignatureMethod.RSA_SHA256, null),
    Collections.singletonList(ref)
);

KeyPairGenerator kpg = KeyPairGenerator.getInstance("RSA");
kpg.initialize(2048);
KeyPair kp = kpg.generateKeyPair();

DOMSignContext dsc = new DOMSignContext(kp.getPrivate(), root); // 插入到root末尾
dsc.setBaseURI("file:///tmp/");

XMLSignature signature = fac.newXMLSignature(si, fac.newKeyInfo(Collections.singletonList(
    fac.newKeyName("my-key")
)));
signature.sign(dsc);

TransformerFactory.newInstance().newTransformer().transform(
    new DOMSource(doc), new StreamResult(new FileOutputStream("signed.xml"))
);

为什么用RSA-SHA256但验证报"algorithm not supported"

不是JDK不支持,而是运行时Security Provider没注册对应算法。Java 8u161+ 默认启用 SunMSCAPISunJCE,但部分定制JRE或容器环境会移除SHA256withRSA支持。

  • 检查是否启用:运行 java -security -version,确认输出含 SunJCE 提供者
  • 手动添加Provider(不推荐生产):Security.addProvider(new com.sun.crypto.provider.SunJCE());
  • 更可靠的做法:签名时明确指定Provider实例,例如 XMLSignatureFactory.getInstance("DOM", new XMLDSigRI())(需引入 xmlsec 库)
  • 避免踩坑:始终用 SignatureMethod.RSA_SHA256 常量,不要拼字符串 "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"——不同JDK对URI解析宽松度不同

XML签名的脆弱点不在密码学,而在DOM树状态、URI解析路径、Canonicalization细节。调试时优先比对签名后XML中 <digestvalue></digestvalue> 和你手动计算的摘要值是否一致,这能快速定位是数据源问题还是算法配置问题。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1946

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2119

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1168

2024.11.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

760

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1566

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

649

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1208

2024.03.22

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号