Java實現AES和RSA演算法
說明:
本文是用 Java1.8 官方的工具類進行的封裝,兩種加密演算法的原理參考:
AES:https://blog.csdn.net/gulang03/article/details/81175854
RSA:https://blog.csdn.net/gulang03/article/details/81176133
實現類:
AESUtil:
package com.fknight.sbsmdemo.tools;
import javax.crypto.*;
import javax.crypto.spec.SecretKeySpec;
import java.io.UnsupportedEncodingException;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Base64;
/**
* AES 加密方法,是對稱的密碼演算法(加密與解密的金鑰一致),這裡使用最大的 256 位的金鑰
*/
public class AESUtil {
/**
* 獲得一個 金鑰長度為 256 位的 AES 金鑰,
* @return 返回經 BASE64 處理之後的金鑰字串
*/
public static String getStrKeyAES() throws NoSuchAlgorithmException, UnsupportedEncodingException {
KeyGenerator keyGen = KeyGenerator.getInstance("AES");
SecureRandom secureRandom = new SecureRandom(String.valueOf(System.currentTimeMillis()).getBytes("utf-8"));
keyGen.init(256, secureRandom); // 這裡可以是 128、192、256、越大越安全
SecretKey secretKey = keyGen.generateKey();
return Base64.getEncoder().encodeToString(secretKey.getEncoded());
}
/**
* 將使用 Base64 加密後的字串型別的 secretKey 轉為 SecretKey
* @param strKey
* @return SecretKey
*/
public static SecretKey strKey2SecretKey(String strKey){
byte[] bytes = Base64.getDecoder().decode(strKey);
SecretKeySpec secretKey = new SecretKeySpec(bytes, "AES");
return secretKey;
}
/**
* 加密
* @param content 待加密內容
* @param secretKey 加密使用的 AES 金鑰
* @return 加密後的密文 byte[]
*/
public static byte[] encryptAES(byte[] content, SecretKey secretKey) throws NoSuchPaddingException, NoSuchAlgorithmException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException {
Cipher cipher = Cipher.getInstance("AES");
cipher.init(Cipher.ENCRYPT_MODE, secretKey);
return cipher.doFinal(content);
}
/**
* 解密
* @param content 待解密內容
* @param secretKey 解密使用的 AES 金鑰
* @return 解密後的明文 byte[]
*/
public static byte[] decryptAES(byte[] content, SecretKey secretKey) throws NoSuchPaddingException, NoSuchAlgorithmException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException {
Cipher cipher = Cipher.getInstance("AES");
cipher.init(Cipher.DECRYPT_MODE, secretKey);
return cipher.doFinal(content);
}
}
RSAUtil:
package com.fknight.sbsmdemo.tools;
import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.NoSuchPaddingException;
import java.io.UnsupportedEncodingException;
import java.security.*;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.Base64;
/**
* RSA 是非對稱的密碼演算法,金鑰分公鑰和私鑰,公鑰用來加密,私鑰用於解密
*/
public class RSAUtil {
/**
* 生成金鑰對:金鑰對中包含公鑰和私鑰
* @return 包含 RSA 公鑰與私鑰的 keyPair
* @throws NoSuchAlgorithmException
* @throws UnsupportedEncodingException
*/
public static KeyPair getKeyPair() throws NoSuchAlgorithmException, UnsupportedEncodingException {
KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance("RSA"); // 獲得RSA金鑰對的生成器例項
SecureRandom secureRandom = new SecureRandom(String.valueOf(System.currentTimeMillis()).getBytes("utf-8")); // 說的一個安全的隨機數
keyPairGenerator.initialize(2048, secureRandom); // 這裡可以是1024、2048 初始化一個金鑰對
KeyPair keyPair = keyPairGenerator.generateKeyPair(); // 獲得金鑰對
return keyPair;
}
/**
* 獲取公鑰 (並進行Base64編碼,返回一個 Base64 編碼後的字串)
* @param keyPair
* @return 返回一個 Base64 編碼後的公鑰字串
*/
public static String getPublicKey(KeyPair keyPair){
PublicKey publicKey = keyPair.getPublic();
byte[] bytes = publicKey.getEncoded();
return Base64.getEncoder().encodeToString(bytes);
}
/**
* 獲取私鑰(並進行Base64編碼,返回一個 Base64 編碼後的字串)
* @param keyPair
* @return 返回一個 Base64 編碼後的私鑰字串
*/
public static String getPrivateKey(KeyPair keyPair){
PrivateKey privateKey = keyPair.getPrivate();
byte[] bytes = privateKey.getEncoded();
return Base64.getEncoder().encodeToString(bytes);
}
/**
* 將Base64編碼後的公鑰轉換成 PublicKey 物件
* @param pubStr
* @return PublicKey
*/
public static PublicKey string2PublicKey(String pubStr) throws NoSuchAlgorithmException, InvalidKeySpecException {
byte[] bytes = Base64.getDecoder().decode(pubStr);
X509EncodedKeySpec keySpec = new X509EncodedKeySpec(bytes);
KeyFactory keyFactory = KeyFactory.getInstance("RSA");
PublicKey publicKey = keyFactory.generatePublic(keySpec);
return publicKey;
}
/**
* 將Base64編碼後的私鑰轉換成 PrivateKey 物件
* @param priStr
* @return PrivateKey
*/
public static PrivateKey string2Privatekey(String priStr) throws NoSuchAlgorithmException, InvalidKeySpecException {
byte[] bytes = Base64.getDecoder().decode(priStr);
PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(bytes);
KeyFactory keyFactory = KeyFactory.getInstance("RSA");
PrivateKey privateKey = keyFactory.generatePrivate(keySpec);
return privateKey;
}
/**
* 公鑰加密
* @param content 待加密的內容 byte[]
* @param publicKey 加密所需的公鑰物件 PublicKey
* @return 加密後的位元組陣列 byte[]
*/
public static byte[] publicEncrytype(byte[] content, PublicKey publicKey) throws NoSuchPaddingException, NoSuchAlgorithmException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException {
Cipher cipher = Cipher.getInstance("RSA");
cipher.init(Cipher.ENCRYPT_MODE, publicKey);
byte[] bytes = cipher.doFinal(content);
return bytes;
}
/**
* 私鑰解密
* @param content 待解密的內容 byte[]
* @param privateKey 解密需要的私鑰物件 PrivateKey
* @return 解密後的位元組陣列 byte[]
*/
public static byte[] privateDecrypt(byte[] content, PrivateKey privateKey) throws NoSuchPaddingException, NoSuchAlgorithmException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException {
Cipher cipher = Cipher.getInstance("RSA");
cipher.init(Cipher.DECRYPT_MODE, privateKey);
byte[] bytes = cipher.doFinal(content);
return bytes;
}
}
測試:
AESUtilTest
package com.fknight.sbsmdemo.tools;
import javax.crypto.SecretKey;
import java.util.Base64;
/**
* 測試 AESUtil 對AES加密演算法的封裝
*/
public class AESUtilTest {
public static void main(String[] args){
String content = "abcdefg789+-*+="; // 待加密的字串
System.out.println("明文資料為:" + content);
try {
// 獲得經 BASE64 處理之後的 AES 金鑰
String strKeyAES = AESUtil.getStrKeyAES();
System.out.println("經BASE64處理之後的金鑰:" + strKeyAES);
// 將 BASE64 處理之後的 AES 金鑰轉為 SecretKey
SecretKey secretKey = AESUtil.strKey2SecretKey(strKeyAES);
// 加密資料
byte[] encryptAESbytes = AESUtil.encryptAES(content.getBytes("utf-8"), secretKey);
System.out.println("加密後的資料經 BASE64 處理之後為:" + Base64.getEncoder().encodeToString(encryptAESbytes));
// 解密資料
String decryptAESStr = new String(AESUtil.decryptAES(encryptAESbytes, secretKey), "utf-8");
System.out.println("解密後的資料為:" + decryptAESStr);
if (content.equals(decryptAESStr)){
System.out.println("測試通過!");
}else {
System.out.println("測試未通過!");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
測試結果:
RSAUtilTest
package com.fknight.sbsmdemo.tools;
import java.security.*;
import java.util.Base64;
/**
* 對 RSAUtil 進行測試
*/
public class RSAUtilTest {
public static void main(String[] args){
String content = "abcdefg456+-="; // 明文內容
System.out.println("原始字串是:" + content);
try {
// 獲得金鑰對
KeyPair keyPair = RSAUtil.getKeyPair();
// 獲得進行Base64 加密後的公鑰和私鑰 String
String privateKeyStr = RSAUtil.getPrivateKey(keyPair);
String publicKeyStr = RSAUtil.getPublicKey(keyPair);
System.out.println("Base64處理後的私鑰:" + privateKeyStr + "\n"
+ "Base64處理後的公鑰:" + publicKeyStr);
// 獲得原始的公鑰和私鑰,並以字串形式列印出來
PrivateKey privateKey = RSAUtil.string2Privatekey(privateKeyStr);
PublicKey publicKey = RSAUtil.string2PublicKey(publicKeyStr);
// 公鑰加密/私鑰解密
byte[] publicEncryBytes = RSAUtil.publicEncrytype(content.getBytes(), publicKey);
System.out.println("公鑰加密後的字串(經BASE64處理):" + Base64.getEncoder().encodeToString(publicEncryBytes));
byte[] privateDecryBytes = RSAUtil.privateDecrypt(publicEncryBytes, privateKey);
System.out.println("私鑰解密後的原始字串:" + new String(privateDecryBytes));
String privateDecryStr = new String(privateDecryBytes, "utf-8");
if (content.equals(privateDecryStr)){
System.out.println("測試通過!");
}else {
System.out.println("測試未通過!");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
測試結果:
PS:關於如何使用 JavaScript 和 Java 進行跨語言 AES 和 RSA 的實現可以參考我的另一篇博文
相關文章
- 【5】JMicro其於RSA及AES加密實現安全服務呼叫加密
- php中aes加密和rsa加密的區別PHP加密
- AES和DES程式碼實現
- RSA演算法與Python實現演算法Python
- nodejs常用加密方式 RSA & AESNodeJS加密
- 安全篇-AES/RSA加密機制加密
- 前後端java+vue 實現rsa 加解密與摘要簽名演算法後端JavaVue解密演算法
- AES演算法測試用例程式Java實現(金鑰長度128位元)演算法Java
- AES 演算法演算法
- RSA加密演算法簡單介紹以及python實現加密演算法Python
- java/php/c#版rsa簽名以及java驗籤實現JavaPHPC#
- STM32: 實現ADVANCED ENCRYPTION STANDARD(AES) – 128-BIT加密演算法加密演算法
- RSA演算法演算法
- 排序演算法Java實現排序演算法Java
- KMP演算法 Java實現KMP演算法Java
- 排序演算法 Java實現排序演算法Java
- Java實現氣泡排序和插入排序演算法Java排序演算法
- RSA加密解密(無資料大小限制,php、go、java互通實現)加密解密PHPGoJava
- 使用OpenSSL替代MCrypt實現AES加解密解密
- RSA der加密 p12解密以及配合AES使用詳解加密解密
- java RSA 解密Java解密
- RSA加密演算法加密演算法
- RSA演算法(一)演算法
- RSA演算法原理演算法
- C#基於RSA加密演算法實現軟體註冊實戰演練C#加密演算法
- JavaScript前端和Java後端的AES加密和解密JavaScript前端後端加密解密
- app直播原始碼,android AES加密解密實現APP原始碼Android加密解密
- 通過Go實現AES加密和解密工具Go加密解密
- [譯]最佳安全實踐:在 Java 和 Android 中使用 AES 進行對稱加密:第2部分:AES-CBC + HMACJavaAndroid加密Mac
- [譯] 最佳安全實踐:在 Java 和 Android 中使用 AES 進行對稱加密JavaAndroid加密
- 排序演算法-Java實現快速排序演算法排序演算法Java
- Java Stream API:實現 Kruskal 演算法JavaAPI演算法
- Java中Blowfish加密演算法實現Java加密演算法
- java實現字元壓縮演算法Java字元演算法
- QQ TEA加密演算法 JAVA實現加密演算法Java
- java氣泡排序演算法實現Java排序演算法
- 基於MD5+RSA演算法實現介面呼叫防扯皮級鑑權演算法
- RSA演算法簡介演算法