当前位置:   article > 正文

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

springboot rsa加密

一、关于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的密钥生成、数据加密解密以及签名和验签操作。

二、代码实现

生成公钥和私钥

  1. import javax.crypto.Cipher;
  2. import java.nio.charset.StandardCharsets;
  3. import java.security.*;
  4. import java.util.Base64;
  5. public class Main {
  6. /**
  7. * RSA算法常量,用于生成RSA密钥对和进行加解密操作。
  8. */
  9. private static final String RSA_ALGORITHM = "RSA";
  10. /**
  11. * 生成RSA密钥对。
  12. *
  13. * @return KeyPair 包含公钥和私钥的密钥对。
  14. * @throws NoSuchAlgorithmException 如果RSA算法不可用,抛出此异常。
  15. */
  16. public static KeyPair generateKeyPair() throws NoSuchAlgorithmException {
  17. // 实例化一个密钥对生成器,指定算法为RSA
  18. KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance(RSA_ALGORITHM);
  19. // 初始化密钥对生成器,指定密钥长度为2048位
  20. keyPairGenerator.initialize(2048); // 密钥大小为2048位
  21. // 生成密钥对
  22. return keyPairGenerator.generateKeyPair();
  23. }
  24. /**
  25. * 使用公钥加密数据。
  26. *
  27. * @param data 待加密的明文数据。
  28. * @param publicKey 公钥,用于加密数据。
  29. * @return String 加密后的数据,以Base64编码表示。
  30. * @throws Exception 如果加密过程中发生错误,抛出此异常。
  31. *
  32. * 此部分代码首先实例化一个Cipher对象,用于执行加密操作。它使用RSA算法,
  33. * 并初始化为加密模式。然后,使用提供的公钥将Cipher对象配置为准备加密状态。
  34. * 最后,将待加密的数据转换为字节数组,进行加密操作,并将加密后的数据
  35. * 使用Base64编码为字符串返回。
  36. */
  37. public static String encrypt(String data, PublicKey publicKey) throws Exception {
  38. Cipher cipher = Cipher.getInstance(RSA_ALGORITHM);
  39. cipher.init(Cipher.ENCRYPT_MODE, publicKey);
  40. byte[] encryptedData = cipher.doFinal(data.getBytes(StandardCharsets.UTF_8));
  41. return Base64.getEncoder().encodeToString(encryptedData);
  42. }
  43. /**
  44. * 使用私钥解密数据。
  45. *
  46. * @param encryptedData 加密后的数据,以Base64编码表示。
  47. * @param privateKey 私钥,用于解密数据。
  48. * @return String 解密后的明文数据。
  49. * @throws Exception 如果解密过程中发生错误,抛出此异常。
  50. */
  51. public static String decrypt(String encryptedData, PrivateKey privateKey) throws Exception {
  52. byte[] decodedData = Base64.getDecoder().decode(encryptedData);
  53. Cipher cipher = Cipher.getInstance(RSA_ALGORITHM);
  54. cipher.init(Cipher.DECRYPT_MODE, privateKey);
  55. byte[] decryptedData = cipher.doFinal(decodedData);
  56. return new String(decryptedData, StandardCharsets.UTF_8);
  57. }
  58. /**
  59. * 程序入口主方法,用于演示RSA加密和解密的流程。
  60. * 生成RSA密钥对,并将公钥和私钥以Base64编码的形式打印出来。
  61. * 使用公钥对明文数据进行加密,然后使用私钥对加密后的数据进行解密,展示加密解密的完整性。
  62. *
  63. * @param args 命令行参数
  64. * @throws Exception 如果密钥生成或加密解密过程中发生错误
  65. */
  66. public static void main(String[] args) throws Exception {
  67. // 生成RSA密钥对
  68. KeyPair keyPair = generateKeyPair();
  69. // 获取公钥
  70. PublicKey publicKey = keyPair.getPublic();
  71. // 获取私钥
  72. PrivateKey privateKey = keyPair.getPrivate();
  73. // 将公钥转换为Base64编码的字符串
  74. String publicKeyBase64 = Base64.getEncoder().encodeToString(publicKey.getEncoded());
  75. // 打印公钥
  76. System.out.println("公钥(Base64编码): \n" + publicKeyBase64 + "\n");
  77. // 将私钥转换为Base64编码的字符串
  78. String privateKeyBase64 = Base64.getEncoder().encodeToString(privateKey.getEncoded());
  79. // 打印私钥
  80. System.out.println("私钥(Base64编码): \n" + privateKeyBase64 + "\n");
  81. // 待加密的明文数据
  82. String data = "学习RSA加密";
  83. // 使用公钥加密明文数据
  84. String encryptedData = encrypt(data, publicKey);
  85. // 打印加密后的数据
  86. System.out.println("加密后的数据:" + encryptedData);
  87. // 使用私钥解密加密后的数据
  88. String decryptedData = decrypt(encryptedData, privateKey);
  89. // 打印解密后的数据
  90. System.out.println("解密后的数据:" + decryptedData);
  91. }
  92. }

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

  1. import java.security.KeyFactory;
  2. import java.security.PrivateKey;
  3. import java.security.PublicKey;
  4. import java.security.spec.PKCS8EncodedKeySpec;
  5. import java.security.spec.X509EncodedKeySpec;
  6. import java.util.Base64;
  7. public class TestRsaKeys {
  8. public static void main(String[] args) throws Exception {
  9. // 替换成你的实际公钥和私钥
  10. String publicKeyPEM = "公钥";
  11. String privateKeyPEM = "私钥";
  12. // 解码公钥
  13. byte[] publicBytes = Base64.getDecoder().decode(publicKeyPEM);
  14. X509EncodedKeySpec publicSpec = new X509EncodedKeySpec(publicBytes);
  15. KeyFactory keyFactory = KeyFactory.getInstance("RSA");
  16. PublicKey publicKey = keyFactory.generatePublic(publicSpec);
  17. // 解码私钥
  18. byte[] privateBytes = Base64.getDecoder().decode(privateKeyPEM);
  19. PKCS8EncodedKeySpec privateSpec = new PKCS8EncodedKeySpec(privateBytes);
  20. PrivateKey privateKey = keyFactory.generatePrivate(privateSpec);
  21. // 打印公钥和私钥
  22. System.out.println("Public Key: " + publicKey);
  23. System.out.println("Private Key: " + privateKey);
  24. }
  25. }

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

  1. security:
  2. rsa:
  3. public-key: 公钥
  4. private-key: 私钥

相关工具类

  1. import org.springframework.stereotype.Component;
  2. import java.security.KeyFactory;
  3. import java.security.PrivateKey;
  4. import java.security.PublicKey;
  5. import java.security.spec.PKCS8EncodedKeySpec;
  6. import java.security.spec.X509EncodedKeySpec;
  7. import java.util.Base64;
  8. import javax.crypto.Cipher;
  9. /**
  10. * RSA工具类,提供RSA加密和解密的功能。
  11. */
  12. @Component
  13. public class RsaUtil {
  14. private final RsaKeyProperties rsaKeyProperties;
  15. private PublicKey publicKey;
  16. private PrivateKey privateKey;
  17. /**
  18. * 构造函数,初始化RSA密钥对。
  19. *
  20. * @param rsaKeyProperties RSA密钥配置属性。
  21. * @throws Exception 如果密钥加载失败。
  22. */
  23. public RsaUtil(RsaKeyProperties rsaKeyProperties) throws Exception {
  24. this.rsaKeyProperties = rsaKeyProperties;
  25. loadKeys();
  26. }
  27. /**
  28. * 加载RSA公钥和私钥。
  29. *
  30. * @throws Exception 如果密钥解码或实例化失败。
  31. */
  32. private void loadKeys() throws Exception {
  33. // 清理公钥字符串中的换行符并解码
  34. String publicKeyPEM = rsaKeyProperties.getPublicKey().replaceAll("\\n", "").replaceAll("\\r", "");
  35. byte[] publicBytes = Base64.getDecoder().decode(publicKeyPEM);
  36. X509EncodedKeySpec publicSpec = new X509EncodedKeySpec(publicBytes);
  37. KeyFactory keyFactory = KeyFactory.getInstance("RSA");
  38. this.publicKey = keyFactory.generatePublic(publicSpec);
  39. // 清理私钥字符串中的换行符并解码
  40. String privateKeyPEM = rsaKeyProperties.getPrivateKey().replaceAll("\\n", "").replaceAll("\\r", "");
  41. byte[] privateBytes = Base64.getDecoder().decode(privateKeyPEM);
  42. PKCS8EncodedKeySpec privateSpec = new PKCS8EncodedKeySpec(privateBytes);
  43. this.privateKey = keyFactory.generatePrivate(privateSpec);
  44. }
  45. /**
  46. * 使用RSA公钥加密数据。
  47. *
  48. * @param data 待加密的数据。
  49. * @return 加密后的数据字符串。
  50. * @throws Exception 如果加密失败。
  51. */
  52. public String encrypt(String data) throws Exception {
  53. Cipher cipher = Cipher.getInstance("RSA");
  54. cipher.init(Cipher.ENCRYPT_MODE, publicKey);
  55. byte[] encryptedData = cipher.doFinal(data.getBytes("UTF-8"));
  56. return Base64.getEncoder().encodeToString(encryptedData);
  57. }
  58. /**
  59. * 使用RSA私钥解密数据。
  60. *
  61. * @param encryptedData 待解密的加密数据字符串。
  62. * @return 解密后的数据字符串。
  63. * @throws Exception 如果解密失败。
  64. */
  65. public String decrypt(String encryptedData) throws Exception {
  66. Cipher cipher = Cipher.getInstance("RSA");
  67. cipher.init(Cipher.DECRYPT_MODE, privateKey);
  68. byte[] decodedData = Base64.getDecoder().decode(encryptedData);
  69. byte[] decryptedData = cipher.doFinal(decodedData);
  70. return new String(decryptedData, "UTF-8");
  71. }
  72. }
  1. import org.springframework.boot.context.properties.ConfigurationProperties;
  2. import org.springframework.context.annotation.Configuration;
  3. /**
  4. * RSA密钥配置类,用于存储和访问RSA公钥和私钥。
  5. * 该类通过@ConfigurationProperties注解绑定到配置文件中以security.rsa为前缀的属性。
  6. */
  7. @Configuration
  8. @ConfigurationProperties(prefix = "security.rsa")
  9. public class RsaKeyProperties {
  10. // 存储RSA公钥
  11. private String publicKey;
  12. // 存储RSA私钥
  13. private String privateKey;
  14. /**
  15. * 获取RSA公钥。
  16. *
  17. * @return RSA公钥字符串
  18. */
  19. public String getPublicKey() {
  20. return publicKey;
  21. }
  22. /**
  23. * 设置RSA公钥。
  24. *
  25. * @param publicKey RSA公钥字符串
  26. */
  27. public void setPublicKey(String publicKey) {
  28. this.publicKey = publicKey;
  29. }
  30. /**
  31. * 获取RSA私钥。
  32. *
  33. * @return RSA私钥字符串
  34. */
  35. public String getPrivateKey() {
  36. return privateKey;
  37. }
  38. /**
  39. * 设置RSA私钥。
  40. *
  41. * @param privateKey RSA私钥字符串
  42. */
  43. public void setPrivateKey(String privateKey) {
  44. this.privateKey = privateKey;
  45. }
  46. }

测试

  1. import org.springframework.beans.factory.annotation.Autowired;
  2. import org.springframework.boot.CommandLineRunner;
  3. import org.springframework.boot.SpringApplication;
  4. import org.springframework.boot.autoconfigure.SpringBootApplication;
  5. @SpringBootApplication
  6. public class SpringBootRsaApplication implements CommandLineRunner {
  7. @Autowired
  8. private RsaUtil rsaUtil;
  9. public static void main(String[] args) {
  10. SpringApplication.run(SpringBootRsaApplication.class, args);
  11. }
  12. @Override
  13. public void run(String... args) throws Exception {
  14. String originalMessage = "测试一下RSA加密方法";
  15. String encryptedMessage = rsaUtil.encrypt(originalMessage);
  16. String decryptedMessage = rsaUtil.decrypt(encryptedMessage);
  17. System.out.println("原文:" + originalMessage);
  18. System.out.println("加密后: " + encryptedMessage);
  19. System.out.println("解密后:" + decryptedMessage);
  20. }
  21. }

运行结果

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/繁依Fanyi0/article/detail/948516
推荐阅读
相关标签
  

闽ICP备14008679号