0

0

Java生成标准PKCS#8/X.509格式RSA密钥对的正确实践

花韻仙語

花韻仙語

发布时间:2026-02-12 14:46:03

|

106人浏览过

|

来源于php中文网

原创

Java生成标准PKCS#8/X.509格式RSA密钥对的正确实践

本文详解java中生成符合国际标准(pkcs#8私钥、x.509公钥)的pem格式rsa密钥对的方法,解决因格式不兼容导致在jsrsasign、devglan等在线工具中验证失败的问题。

在Java中直接使用KeyPair.getPrivate().getEncoded()和KeyPair.getPublic().getEncoded()拼接PEM头尾(如-----BEGIN RSA PRIVATE KEY-----)所生成的密钥,并非标准PEM格式,而是旧版PKCS#1格式(对应RSAPrivateKey ASN.1结构)。而现代工具(如jsrsasignDevGlan、OpenSSL、JWT库及大多数Web Crypto API)均要求:

  • 私钥必须为PKCS#8格式(-----BEGIN PRIVATE KEY-----),封装PrivateKeyInfo结构;
  • 公钥必须为X.509格式(-----BEGIN PUBLIC KEY-----),封装SubjectPublicKeyInfo结构;
  • ❌ -----BEGIN RSA PRIVATE KEY----- 和 -----BEGIN RSA PUBLIC KEY----- 属于已淘汰的PKCS#1变体,不被主流工具支持

为什么原代码生成的密钥无效?

原始代码中:

sbPrivate.append("-----BEGIN RSA PRIVATE KEY-----\n");
sbPrivate.append(Base64.getEncoder().encodeToString(pair.getPrivate().getEncoded()));

调用的是RSAPrivateKey.getEncoded(),返回的是纯PKCS#1 DER字节(RSAPrivateKey ASN.1序列),而非PKCS#8的PrivateKeyInfo。同理,RSAPublicKey.getEncoded()返回的是PKCS#1公钥,而非X.509标准公钥。二者虽可被部分旧系统识别,但无法通过RFC 5208 / RFC 5280校验,故在线工具报“invalid key”。

正确做法:使用Bouncy Castle生成标准PEM

Java标准库不提供原生PKCS#8/X.509 PEM序列化能力,需借助Bouncy Castle(业界事实标准密码学扩展库)。以下是完整、可运行的解决方案:

传声港
传声港

AI驱动的综合媒体服务平台,提供 “媒体发稿 + 自媒体宣发 + 效果监测” 一站式服务

下载

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

✅ 依赖配置(Maven)

<dependency>
    <groupId>org.bouncycastle</groupId>
    <artifactId>bcpkix-jdk15on</artifactId>
    <version>1.70</version> <!-- 推荐使用1.70+,兼容JDK 11+ -->
</dependency>
⚠️ 注意:bcpkix-jdk15on 已包含 bcprov-jdk15on,无需重复引入。

✅ 标准PEM生成代码

import org.bouncycastle.openssl.jcajce.JcaPEMWriter;
import org.bouncycastle.util.io.pem.PemObject;
import org.bouncycastle.util.io.pem.PemWriter;

import java.io.StringWriter;
import java.security.KeyPair;
import java.security.KeyPairGenerator;

public class StandardRsaPemGenerator {

    public static String generateStandardPem() throws Exception {
        // 1. 生成2048位RSA密钥对
        KeyPairGenerator kpg = KeyPairGenerator.getInstance("RSA");
        kpg.initialize(2048);
        KeyPair kp = kpg.generateKeyPair();

        // 2. 使用Bouncy Castle写入标准PEM
        StringWriter writer = new StringWriter();

        // 私钥:PKCS#8格式(-----BEGIN PRIVATE KEY-----)
        try (JcaPEMWriter pemWriter = new JcaPEMWriter(writer)) {
            pemWriter.writeObject(kp.getPrivate()); // 自动识别并封装为PKCS#8
        }

        // 公钥:X.509格式(-----BEGIN PUBLIC KEY-----)
        try (PemWriter pemWriter = new PemWriter(writer)) {
            pemWriter.writeObject(new PemObject("PUBLIC KEY", kp.getPublic().getEncoded()));
        }

        return writer.toString();
    }

    // 示例调用
    public static void main(String[] args) throws Exception {
        System.out.println(generateStandardPem());
    }
}

✅ 输出示例(标准格式)

-----BEGIN PRIVATE KEY-----
MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQC...
-----END PRIVATE KEY-----
-----BEGIN PUBLIC KEY-----
MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAt...
-----END PUBLIC KEY-----

✅ 验证方式:

  • 访问 8gwifi.org PEM Parser → 粘贴私钥/公钥 → 显示 Private Format: PKCS#8 / Format: X.509 即为成功;
  • jsrsasign demo 中可正常加载、签名/验签;
  • OpenSSL命令行验证:
    openssl rsa -in private.pem -check -noout      # 检查PKCS#8私钥
    openssl rsa -pubin -in public.pem -text -noout # 检查X.509公钥

⚠️ 关键注意事项

  • 不要手动拼接PEM头尾:getEncoded()返回的是DER字节,其ASN.1结构决定格式类型,头尾标签仅是提示,不能改变实质;
  • 避免使用过时的JcaPKCS8Generator显式构造(如答案中旧示例):Bouncy Castle 1.60+ 后,JcaPEMWriter.writeObject(PrivateKey) 会自动选择最优封装,更简洁可靠;
  • 公钥必须用"PUBLIC KEY"类型名:不可写作"RSA PUBLIC KEY",否则解析为PKCS#1而非X.509;
  • 确保Bouncy Castle Provider已注册(若遇NoSuchAlgorithmException):
    Security.addProvider(new BouncyCastleProvider());
  • 生产环境建议指定安全随机源
    kpg.initialize(2048, new SecureRandom()); // 避免默认弱熵

总结

生成可互操作的RSA PEM密钥,核心在于遵循标准编码规范,而非简单Base64封装。Java原生API仅提供底层密钥材料,需借助Bouncy Castle完成PKCS#8/X.509的语义封装。采用本文方案,即可确保密钥在浏览器Web Crypto、Node.js crypto、OpenSSL、JWT服务及所有符合RFC标准的系统中无缝使用。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
Java Maven专题
Java Maven专题

本专题聚焦 Java 主流构建工具 Maven 的学习与应用,系统讲解项目结构、依赖管理、插件使用、生命周期与多模块项目配置。通过企业管理系统、Web 应用与微服务项目实战,帮助学员全面掌握 Maven 在 Java 项目构建与团队协作中的核心技能。

0

2025.09.15

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

865

2023.07.31

python中的format是什么意思
python中的format是什么意思

python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

444

2024.06.27

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

520

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

350

2023.07.28

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

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

508

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

5557

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

487

2023.09.01

2026春节习俗大全
2026春节习俗大全

本专题整合了2026春节习俗大全,阅读专题下面的文章了解更多详细内容。

189

2026.02.11

热门下载

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

精品课程

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

共23课时 | 3.5万人学习

C# 教程
C# 教程

共94课时 | 9.3万人学习

Java 教程
Java 教程

共578课时 | 64.1万人学习

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

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