【SpringBoot】RSA加密(非对称加密)

news2024/9/21 12:37:12

一、关于RSA

        RSA是一种非对称加密算法,广泛应用于数据加密和数字签名领域。

        RSA算法是由罗纳德·李维斯特(Ron Rivest)、阿迪·萨莫尔(Adi Shamir)和伦纳德·阿德曼(Leonard Adleman)在1977年提出的。该算法基于一个十分简单的数论事实:将两个大素数相乘得到它们的乘积很容易,但反过来,已知乘积压根求解出这两个素数却极其困难。这种特性使得RSA能够有效地进行公钥加密和私钥解密。

        在Java中,实现RSA算法需要生成一对密钥,即公钥和私钥。公钥用于加密数据,而私钥用于解密数据。具体来说,当数据被公钥加密后,只能通过相应的私钥解密;反之亦然,当数据被私钥加密后,只能通过相应的公钥解密。这种设计确保了数据的安全性,因为即使公钥被公开,没有私钥,数据也无法被破解。

        在Java中,可以使用`java.security.KeyPairGenerator`类来生成RSA密钥对。通常,密钥的长度为2048位,这是目前推荐的密钥长度,因为它提供了足够的安全性。生成的公钥和私钥可以转换成Base64编码的字符串,方便存储和传输。

        除了加密和解密功能外,RSA还常用于数字签名和验签。私钥可以用来签署数据,生成签名,而公钥则用来验证这个签名的真实性。这一功能在电子商务和网络通信中尤为重要,它保证了数据的完整性和不可抵赖性。例如,在网上支付过程中,用户通过私钥对支付信息进行签名,支付平台使用用户的公钥验证签名的真实性,从而确认交易是用户本人发起的。

        RSA算法也被广泛应用于HTTPS协议中,以保障互联网数据传输的安全性。在HTTPS协议中,服务器使用RSA算法的公钥加密传输给客户端的数据,客户端使用对应的私钥解密数据,确保数据在传输过程中不被第三方窥视或篡改。

        RSA算法作为一种非对称加密技术,不仅提供了高安全性的数据保护,还在数字签名方面有着广泛的应用。在Java环境中,通过各种工具类和库函数,可以便捷地实现RSA的密钥生成、数据加密解密以及签名和验签操作。

二、代码实现

生成公钥和私钥

import javax.crypto.Cipher;
import java.nio.charset.StandardCharsets;
import java.security.*;
import java.util.Base64;

public class Main {
    /**
     * RSA算法常量,用于生成RSA密钥对和进行加解密操作。
     */
    private static final String RSA_ALGORITHM = "RSA";

    /**
     * 生成RSA密钥对。
     *
     * @return KeyPair 包含公钥和私钥的密钥对。
     * @throws NoSuchAlgorithmException 如果RSA算法不可用,抛出此异常。
     */
    public static KeyPair generateKeyPair() throws NoSuchAlgorithmException {
        // 实例化一个密钥对生成器,指定算法为RSA
        KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance(RSA_ALGORITHM);
        // 初始化密钥对生成器,指定密钥长度为2048位
        keyPairGenerator.initialize(2048); // 密钥大小为2048位
        // 生成密钥对
        return keyPairGenerator.generateKeyPair();

    }

    /**
     * 使用公钥加密数据。
     *
     * @param data 待加密的明文数据。
     * @param publicKey 公钥,用于加密数据。
     * @return String 加密后的数据,以Base64编码表示。
     * @throws Exception 如果加密过程中发生错误,抛出此异常。
     *
     * 此部分代码首先实例化一个Cipher对象,用于执行加密操作。它使用RSA算法,
     * 并初始化为加密模式。然后,使用提供的公钥将Cipher对象配置为准备加密状态。
     * 最后,将待加密的数据转换为字节数组,进行加密操作,并将加密后的数据
     * 使用Base64编码为字符串返回。
     */
    public static String encrypt(String data, PublicKey publicKey) throws Exception {
        Cipher cipher = Cipher.getInstance(RSA_ALGORITHM);
        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
        byte[] encryptedData = cipher.doFinal(data.getBytes(StandardCharsets.UTF_8));
        return Base64.getEncoder().encodeToString(encryptedData);
    }

    /**
     * 使用私钥解密数据。
     *
     * @param encryptedData 加密后的数据,以Base64编码表示。
     * @param privateKey 私钥,用于解密数据。
     * @return String 解密后的明文数据。
     * @throws Exception 如果解密过程中发生错误,抛出此异常。
     */
    public static String decrypt(String encryptedData, PrivateKey privateKey) throws Exception {
        byte[] decodedData = Base64.getDecoder().decode(encryptedData);
        Cipher cipher = Cipher.getInstance(RSA_ALGORITHM);
        cipher.init(Cipher.DECRYPT_MODE, privateKey);
        byte[] decryptedData = cipher.doFinal(decodedData);
        return new String(decryptedData, StandardCharsets.UTF_8);
    }

    /**
     * 程序入口主方法,用于演示RSA加密和解密的流程。
     * 生成RSA密钥对,并将公钥和私钥以Base64编码的形式打印出来。
     * 使用公钥对明文数据进行加密,然后使用私钥对加密后的数据进行解密,展示加密解密的完整性。
     *
     * @param args 命令行参数
     * @throws Exception 如果密钥生成或加密解密过程中发生错误
     */
    public static void main(String[] args) throws Exception {
        // 生成RSA密钥对
        KeyPair keyPair = generateKeyPair();
        // 获取公钥
        PublicKey publicKey = keyPair.getPublic();
        // 获取私钥
        PrivateKey privateKey = keyPair.getPrivate();

        // 将公钥转换为Base64编码的字符串
        String publicKeyBase64 = Base64.getEncoder().encodeToString(publicKey.getEncoded());
        // 打印公钥
        System.out.println("公钥(Base64编码): \n" + publicKeyBase64 + "\n");

        // 将私钥转换为Base64编码的字符串
        String privateKeyBase64 = Base64.getEncoder().encodeToString(privateKey.getEncoded());
        // 打印私钥
        System.out.println("私钥(Base64编码): \n" + privateKeyBase64 + "\n");

        // 待加密的明文数据
        String data = "学习RSA加密";

        // 使用公钥加密明文数据
        String encryptedData = encrypt(data, publicKey);
        // 打印加密后的数据
        System.out.println("加密后的数据:" + encryptedData);

        // 使用私钥解密加密后的数据
        String decryptedData = decrypt(encryptedData, privateKey);
        // 打印解密后的数据
        System.out.println("解密后的数据:" + decryptedData);
    }

}

测试生成的公钥和私钥是否能用

import java.security.KeyFactory;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.Base64;

public class TestRsaKeys {
    public static void main(String[] args) throws Exception {
        // 替换成你的实际公钥和私钥
        String publicKeyPEM = "公钥";
        String privateKeyPEM = "私钥";

        // 解码公钥
        byte[] publicBytes = Base64.getDecoder().decode(publicKeyPEM);
        X509EncodedKeySpec publicSpec = new X509EncodedKeySpec(publicBytes);
        KeyFactory keyFactory = KeyFactory.getInstance("RSA");
        PublicKey publicKey = keyFactory.generatePublic(publicSpec);

        // 解码私钥
        byte[] privateBytes = Base64.getDecoder().decode(privateKeyPEM);
        PKCS8EncodedKeySpec privateSpec = new PKCS8EncodedKeySpec(privateBytes);
        PrivateKey privateKey = keyFactory.generatePrivate(privateSpec);

        // 打印公钥和私钥
        System.out.println("Public Key: " + publicKey);
        System.out.println("Private Key: " + privateKey);
    }
}

将生成的公钥和私钥保存至yml文件

security:
  rsa:
    public-key: 公钥
    private-key: 私钥

相关工具类

import org.springframework.stereotype.Component;
import java.security.KeyFactory;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.Base64;
import javax.crypto.Cipher;

/**
 * RSA工具类,提供RSA加密和解密的功能。
 */
@Component
public class RsaUtil {
    private final RsaKeyProperties rsaKeyProperties;
    private PublicKey publicKey;
    private PrivateKey privateKey;

    /**
     * 构造函数,初始化RSA密钥对。
     *
     * @param rsaKeyProperties RSA密钥配置属性。
     * @throws Exception 如果密钥加载失败。
     */
    public RsaUtil(RsaKeyProperties rsaKeyProperties) throws Exception {
        this.rsaKeyProperties = rsaKeyProperties;
        loadKeys();
    }

    /**
     * 加载RSA公钥和私钥。
     *
     * @throws Exception 如果密钥解码或实例化失败。
     */
    private void loadKeys() throws Exception {
        // 清理公钥字符串中的换行符并解码
        String publicKeyPEM = rsaKeyProperties.getPublicKey().replaceAll("\\n", "").replaceAll("\\r", "");
        byte[] publicBytes = Base64.getDecoder().decode(publicKeyPEM);
        X509EncodedKeySpec publicSpec = new X509EncodedKeySpec(publicBytes);
        KeyFactory keyFactory = KeyFactory.getInstance("RSA");
        this.publicKey = keyFactory.generatePublic(publicSpec);

        // 清理私钥字符串中的换行符并解码
        String privateKeyPEM = rsaKeyProperties.getPrivateKey().replaceAll("\\n", "").replaceAll("\\r", "");
        byte[] privateBytes = Base64.getDecoder().decode(privateKeyPEM);
        PKCS8EncodedKeySpec privateSpec = new PKCS8EncodedKeySpec(privateBytes);
        this.privateKey = keyFactory.generatePrivate(privateSpec);
    }

    /**
     * 使用RSA公钥加密数据。
     *
     * @param data 待加密的数据。
     * @return 加密后的数据字符串。
     * @throws Exception 如果加密失败。
     */
    public String encrypt(String data) throws Exception {
        Cipher cipher = Cipher.getInstance("RSA");
        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
        byte[] encryptedData = cipher.doFinal(data.getBytes("UTF-8"));
        return Base64.getEncoder().encodeToString(encryptedData);
    }

    /**
     * 使用RSA私钥解密数据。
     *
     * @param encryptedData 待解密的加密数据字符串。
     * @return 解密后的数据字符串。
     * @throws Exception 如果解密失败。
     */
    public String decrypt(String encryptedData) throws Exception {
        Cipher cipher = Cipher.getInstance("RSA");
        cipher.init(Cipher.DECRYPT_MODE, privateKey);
        byte[] decodedData = Base64.getDecoder().decode(encryptedData);
        byte[] decryptedData = cipher.doFinal(decodedData);
        return new String(decryptedData, "UTF-8");
    }

}
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;

/**
 * RSA密钥配置类,用于存储和访问RSA公钥和私钥。
 * 该类通过@ConfigurationProperties注解绑定到配置文件中以security.rsa为前缀的属性。
 */
@Configuration
@ConfigurationProperties(prefix = "security.rsa")
public class RsaKeyProperties {
    // 存储RSA公钥
    private String publicKey;
    // 存储RSA私钥
    private String privateKey;

    /**
     * 获取RSA公钥。
     *
     * @return RSA公钥字符串
     */
    public String getPublicKey() {
        return publicKey;
    }

    /**
     * 设置RSA公钥。
     *
     * @param publicKey RSA公钥字符串
     */
    public void setPublicKey(String publicKey) {
        this.publicKey = publicKey;
    }

    /**
     * 获取RSA私钥。
     *
     * @return RSA私钥字符串
     */
    public String getPrivateKey() {
        return privateKey;
    }

    /**
     * 设置RSA私钥。
     *
     * @param privateKey RSA私钥字符串
     */
    public void setPrivateKey(String privateKey) {
        this.privateKey = privateKey;
    }

}

测试

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class SpringBootRsaApplication implements CommandLineRunner {

    @Autowired
    private RsaUtil rsaUtil;

    public static void main(String[] args) {
        SpringApplication.run(SpringBootRsaApplication.class, args);
    }

    @Override
    public void run(String... args) throws Exception {
        String originalMessage = "测试一下RSA加密方法";
        String encryptedMessage = rsaUtil.encrypt(originalMessage);
        String decryptedMessage = rsaUtil.decrypt(encryptedMessage);

        System.out.println("原文:" + originalMessage);
        System.out.println("加密后: " + encryptedMessage);
        System.out.println("解密后:" + decryptedMessage);
    }

}

运行结果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1840124.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JavaScript事件传播实战

上篇文章我们学习了事件传播的冒泡和捕获两种类型,现在我们在实际项目中演示一下; ● 首先我们先定义一个随机数 const randomInt (min, max) > Math.floor(Math.random() * (max - min 1) min);● 接着,我们使用随机数来创建随机的r…

osi七层参考模型和tcp/ip模型的区别与相似之处

osi七层参考模型: 2.tcp/ip四层参考模型: osi七层参考模型与tcp/ip四层参考模型的相似与区别: 相同点: 2者都是模型化层次化 下层对上层提供服务支持 每层协议彼此相互独立 不同点:OSI先有模型才有协议 TCP/IP先有…

目标检测数据集 - 零售食品LOGO检测数据集下载「包含VOC、COCO、YOLO三种格式」

数据集介绍:零售食品 LOGO 检测数据集,真实零售食品 LOGO 高质量商品图片数据,数据集含常见零售食品 LOGO 图片,包括饮料类、酒类、调味品类、膨化饼干类、巧克力类、常见零食类等等。数据集类别丰富,标注标签包含 150…

【因果推断python】44_评估因果模型2

目录 累积弹性曲线 累积增益曲线 考虑差异 关键思想 累积弹性曲线 再次考虑将价格转换为二元处理的说明性示例。我们会从我们离开的地方拿走它,所以我们有弹性处理带。我们接下来可以做的是根据乐队的敏感程度对乐队进行排序。也就是说,我们把最敏感…

Python学习打卡:day10

day10 笔记来源于:黑马程序员python教程,8天python从入门到精通,学python看这套就够了 目录 day1073、文件的读取操作文件的操作步骤open()打开函数mode常用的三种基础访问模式读操作相关方法read()方法readlines()方法readline()方法for循…

大模型微调和RAG的应用场景

大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算法…

进入docker容器内部操作mysql数据库

文章目录 1、查询docker容器2、进入mysql容器内部3、连接mysql数据库4、查询mysql所有的数据库5、使用某个数据库6、展示数据库中所有的表7、查询某张表8、断开mysql9、退出mysql容器 1、查询docker容器 [rootlocalhost ~]# docker ps CONTAINER ID IMAGE …

聚观早报 | 苹果AI功能将分批上线;三星Galaxy Z Fold6尺寸数据

聚观早报每日整理最值得关注的行业重点事件,帮助大家及时了解最新行业动态,每日读报,就读聚观365资讯简报。 整理丨Cutie 6月18日消息 苹果AI功能将分批上线 三星Galaxy Z Fold6尺寸数据 华为智慧PC新品AI能力升级 抖音出品《三星堆&am…

Spring MVC学习记录(基础)

目录 1.SpringMVC概述1.1 MVC介绍1.2 Spring MVC介绍1.3 Spring MVC 的核心组件1.4 SpringMVC 工作原理 2.Spring MVC入门2.1 入门案例2.2 总结 3.RequestMapping注解4.controller方法返回值4.1 返回ModelAndView4.2 返回字符串4.2.1 逻辑视图名4.2.2 Redirect重定向4.2.3 forw…

六面体大米装袋机长期稳定运行原因分析

随着现代化农业生产的发展,六面体大米装袋机已成为粮食加工行业不可或缺的重要设备。然而,如何确保这些机器长期稳定运行,提高生产效率,降低维护成本,一直是广大粮食加工企业关注的焦点。星派将为您揭示六面体大米装袋…

如何在前端项目中用字体图标替换图片,方便减小打包体积和统一切换颜色

1.进入阿里妈妈矢量图标图库 地址:阿里妈妈矢量图 2.搜索自己想要的图标 3.添加自己想要的图标 4.把刚才选的图标,添加到自己要下载的项目 5.把项目下载到本地 6.引入iconfont.css 在div上增加对应的类名就可以啦 下载的所有类名都在下面的demo_index…

测速小车模块

1.用途:广泛用于电机转速检测,脉冲计数,位置限位等。 2.有遮挡,输出高电平;无遮挡,输出低电平 接线 VCC 接电源正极3.3-5V GND 接电源负极 DO TTL开关信号输出 AO 此模块不起作用 测试原理和单位换算:…

如何定制Spring的错误json信息

一,前言 相信很多同学都有遇到过这样的spring错误信息。 在我们没有做catch处理时或者做全局的exceptionHandle时,Spring遇到抛出向外的异常时,就会给我们封装返回这么个格式的异常信息。 那么问题来了,我们能否对这个返回增加错…

[Linux] vi编辑器

命令模式&文本模式 命令模式就输入命令然后执行,文本模式就是系统把你的输入都当成写进文件里的字符 切换模式: 刚进入默认是命令模式,按: i I a A o O 进入文本模式, 通过他们进入文本模式有什么不同? 然后按esc进…

PLC模拟量和数字量到底有什么区别?

PLC模拟量和数字量的区别 在工业自动化领域,可编程逻辑控制器(PLC)是控制各种机械设备和生产过程的核心组件。PLC通过处理模拟量和数字量来实现对工业过程的精确控制。了解模拟量和数字量的区别对于设计高效、可靠的自动化系统至关重要。 1. …

vue3+ts 使用vue3-ace-editor实现Json编辑器

1、效果图 输入代码,点击格式化就出现以上效果,再点击压缩,是以下效果2、安装 npm i vue3-ace-editor 3、使用 新建aceConfig.js文件 // ace配置,使用动态加载来避免第一次加载开销 import ace from ace-builds// 导入不同的主…

查看服务器接口各行代码耗时的 linux 命令

如何通过命令来查看接口具体位置的耗时情况 首先然后找到需要执行查看耗时的方法执行查询命令 trace 首先 使用这个命令 curl -O https://arthas.aliyun.com/arthas-boot.jar java -jar arthas-boot.jar然后找到需要执行查看耗时的方法 复制出来的 是 com.a.tms.api.service.…

昇思MindSpore全场景深度学习框架总结

昇思MindSpore介绍 MindSpore是一个全场景深度学习框架,旨在实现易开发、高效执行、全场景统一部署三大目标,具体包括API友好、调试难度低、计算效率、数据预处理效率和分布式训练效率高以及支持云、边缘和端侧场景。 昇思MindSpore的各个扩展功能模块&…

基于JSP的高校毕业生就业满意度调查统计系统

开头语: 你好呀,我是计算机学长猫哥!如果有相关需求,文末可以找到我的联系方式。 开发语言:JSP 数据库:MySQL 技术:JSP技术 工具:MyEclipse、Tomcat、MySQL 系统展示 首页 用…