JAVA加密解密之3DES(TripleDES)

x33g5p2x  于2021-12-25 转载在 其他  
字(11.6k)|赞(0)|评价(0)|浏览(544)

3DES算法简介

3DES(或称为Triple DES)是三重数据加密算法(TDEA,Triple Data Encryption Algorithm)块密码的通称。它相当于是对每个数据块应用三次DES加密算法。由于计算机运算能力的增强,原版DES密码的密钥长度变得容易被暴力破解;3DES即是设计用来提供一种相对简单的方法,即通过增加DES的密钥长度来避免类似的攻击,而不是设计一种全新的块密码算法。

3DES算法实现

  1. package com.jianggujin.codec;
  2. import java.io.InputStream;
  3. import java.io.OutputStream;
  4. import javax.crypto.Cipher;
  5. import javax.crypto.KeyGenerator;
  6. import javax.crypto.SecretKey;
  7. import javax.crypto.spec.IvParameterSpec;
  8. import javax.crypto.spec.SecretKeySpec;
  9. import com.jianggujin.codec.util.JCipherInputStream;
  10. import com.jianggujin.codec.util.JCipherOutputStream;
  11. import com.jianggujin.codec.util.JCodecException;
  12. /** * 3DES(或称为Triple DES)是三重数据加密算法(TDEA,Triple Data Encryption * Algorithm)块密码的通称。它相当于是对每个数据块应用三次DES加密算法 * * @author jianggujin * */
  13. public class JTripleDES {
  14. /** * DES算法 * * @author jianggujin * */
  15. public static enum JDESAlgorithm {
  16. DES, DESede;
  17. public String getName() {
  18. return this.name();
  19. }
  20. }
  21. /** * 工作模式 * * @author jianggujin * */
  22. public static enum JWorkingMode {
  23. /** * 该模式不能使用向量 */
  24. ECB, CBC, CFB, OFB, CTR;
  25. public String getName() {
  26. return this.name();
  27. }
  28. }
  29. /** * 填充方式 * * @author jianggujin * */
  30. public static enum JPadding {
  31. NoPadding, PKCS5Padding;
  32. public String getName() {
  33. return this.name();
  34. }
  35. }
  36. /** * 初始化密钥 * * @param algorithm * @return * @throws Exception */
  37. public static byte[] initKey(String algorithm) {
  38. try {
  39. KeyGenerator kg = KeyGenerator.getInstance(algorithm);
  40. return kg.generateKey().getEncoded();
  41. } catch (Exception e) {
  42. throw new JCodecException(e);
  43. }
  44. }
  45. /** * 初始化向量 * * @param algorithm * @param workingMode * @param padding * @return */
  46. public static byte[] initIv(String algorithm, String workingMode, String padding) {
  47. try {
  48. String fullAlg = algorithm + "/" + workingMode + "/" + padding;
  49. Cipher cipher = Cipher.getInstance(fullAlg);
  50. int blockSize = cipher.getBlockSize();
  51. byte[] iv = new byte[blockSize];
  52. for (int i = 0; i < blockSize; i++) {
  53. iv[i] = 0;
  54. }
  55. return iv;
  56. } catch (Exception e) {
  57. throw new JCodecException(e);
  58. }
  59. }
  60. /** * 加密 * * @param data * @param key * @param algorithm * @param workingMode * @param padding * @return */
  61. public static byte[] encrypt(byte[] data, byte[] key, String algorithm, String workingMode, String padding) {
  62. return encrypt(data, key, algorithm, workingMode, padding, null);
  63. }
  64. /** * 加密 * * @param data * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @return */
  65. public static byte[] encrypt(byte[] data, byte[] key, String algorithm, String workingMode, String padding,
  66. byte[] iv) {
  67. // 数据加密
  68. Cipher cipher = getEncryptCipher(key, algorithm, workingMode, padding, iv);
  69. try {
  70. return cipher.doFinal(data);
  71. } catch (Exception e) {
  72. throw new JCodecException(e);
  73. }
  74. }
  75. /** * 包裹输出流,包裹后的输出流为加密输出流 * * @param out * @param key * @param algorithm * @param workingMode * @param padding * @return */
  76. public static OutputStream wrap(OutputStream out, byte[] key, String algorithm, String workingMode, String padding) {
  77. return wrap(out, key, algorithm, workingMode, padding, null);
  78. }
  79. /** * 包裹输出流,包裹后的输出流为加密输出流 * * @param out * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @return */
  80. public static OutputStream wrap(OutputStream out, byte[] key, String algorithm, String workingMode, String padding,
  81. byte[] iv) {
  82. // 数据加密
  83. Cipher cipher = getEncryptCipher(key, algorithm, workingMode, padding, iv);
  84. return new JCipherOutputStream(cipher, out);
  85. }
  86. /** * 获得加密模式的{@link Cipher} * * @param key * @param algorithm * @param workingMode * @param padding * @return */
  87. public static Cipher getEncryptCipher(byte[] key, String algorithm, String workingMode, String padding) {
  88. return getEncryptCipher(key, algorithm, workingMode, padding, null);
  89. }
  90. /** * 获得加密模式的{@link Cipher} * * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @return */
  91. public static Cipher getEncryptCipher(byte[] key, String algorithm, String workingMode, String padding, byte[] iv) {
  92. return getCipher(key, algorithm, workingMode, padding, iv, Cipher.ENCRYPT_MODE);
  93. }
  94. /** * 解密 * * @param data * @param key * @param algorithm * @param workingMode * @param padding * @return */
  95. public static byte[] decrypt(byte[] data, byte[] key, String algorithm, String workingMode, String padding) {
  96. return decrypt(data, key, algorithm, workingMode, padding, null);
  97. }
  98. /** * 解密 * * @param data * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @return */
  99. public static byte[] decrypt(byte[] data, byte[] key, String algorithm, String workingMode, String padding,
  100. byte[] iv) {
  101. // 数据解密
  102. Cipher cipher = getDecryptCipher(key, algorithm, workingMode, padding, iv);
  103. try {
  104. return cipher.doFinal(data);
  105. } catch (Exception e) {
  106. throw new JCodecException(e);
  107. }
  108. }
  109. /** * 包裹输入流,原输入流为加密数据输入流 * * @param in * @param key * @param algorithm * @param workingMode * @param padding * @return */
  110. public static InputStream wrap(InputStream in, byte[] key, String algorithm, String workingMode, String padding) {
  111. return wrap(in, key, algorithm, workingMode, padding, null);
  112. }
  113. /** * 包裹输入流,原输入流为加密数据输入流 * * @param in * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @return */
  114. public static InputStream wrap(InputStream in, byte[] key, String algorithm, String workingMode, String padding,
  115. byte[] iv) {
  116. // 数据解密
  117. Cipher cipher = getDecryptCipher(key, algorithm, workingMode, padding, iv);
  118. return new JCipherInputStream(cipher, in);
  119. }
  120. /** * 获得解密模式的{@link Cipher} * * @param key * @param algorithm * @param workingMode * @param padding * @return */
  121. public static Cipher getDecryptCipher(byte[] key, String algorithm, String workingMode, String padding) {
  122. return getDecryptCipher(key, algorithm, workingMode, padding, null);
  123. }
  124. /** * 获得解密模式的{@link Cipher} * * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @return */
  125. public static Cipher getDecryptCipher(byte[] key, String algorithm, String workingMode, String padding, byte[] iv) {
  126. return getCipher(key, algorithm, workingMode, padding, iv, Cipher.DECRYPT_MODE);
  127. }
  128. /** * 获得{@link Cipher} * * @param key * @param algorithm * @param workingMode * @param padding * @param iv * 为null时则使用默认向量 * @param opmode * @return */
  129. private static Cipher getCipher(byte[] key, String algorithm, String workingMode, String padding, byte[] iv,
  130. int opmode) {
  131. // 生成本地密钥
  132. SecretKey secretKey = getSecretKey(key, algorithm);
  133. try {
  134. String fullAlg = algorithm + "/" + workingMode + "/" + padding;
  135. Cipher cipher = Cipher.getInstance(fullAlg);
  136. if (!JWorkingMode.ECB.getName().equalsIgnoreCase(workingMode)) {
  137. if (iv == null) {
  138. int blockSize = cipher.getBlockSize();
  139. iv = new byte[blockSize];
  140. for (int i = 0; i < blockSize; i++) {
  141. iv[i] = 0;
  142. }
  143. }
  144. IvParameterSpec ivSpec = new IvParameterSpec(iv);
  145. cipher.init(opmode, secretKey, ivSpec);
  146. } else {
  147. cipher.init(opmode, secretKey);
  148. }
  149. return cipher;
  150. } catch (Exception e) {
  151. throw new JCodecException(e);
  152. }
  153. }
  154. /** * 获得密钥 * * @param key * @param algorithm * @return */
  155. private static SecretKey getSecretKey(byte[] key, String algorithm) {
  156. try {
  157. SecretKey secretKey = new SecretKeySpec(key, algorithm);
  158. return secretKey;
  159. } catch (Exception e) {
  160. throw new JCodecException(e);
  161. }
  162. }
  163. }

测试代码:

  1. package com.jianggujin.codec.test;
  2. import java.io.File;
  3. import java.io.FileInputStream;
  4. import java.io.FileOutputStream;
  5. import java.io.InputStream;
  6. import java.io.OutputStream;
  7. import org.junit.Test;
  8. import com.jianggujin.codec.JBase64;
  9. import com.jianggujin.codec.JBase64.JEncoder;
  10. import com.jianggujin.codec.JTripleDES;
  11. import com.jianggujin.codec.JTripleDES.JDESAlgorithm;
  12. import com.jianggujin.codec.JTripleDES.JPadding;
  13. import com.jianggujin.codec.JTripleDES.JWorkingMode;
  14. public class TripleDESTest {
  15. String str = "jianggujin111111";
  16. File file = new File(getClass().getSimpleName() + ".dat");
  17. @Test
  18. public void test() throws Exception {
  19. System.out.println("原串:" + str);
  20. JEncoder encoder = JBase64.getEncoder();
  21. for (JDESAlgorithm algorithm : JDESAlgorithm.values()) {
  22. System.out.println("算法:" + algorithm.getName());
  23. byte[] key = JTripleDES.initKey(algorithm.getName());
  24. System.out.println("密钥:" + encoder.encodeToString(key, "UTF-8"));
  25. for (JWorkingMode workingMode : JWorkingMode.values()) {
  26. for (JPadding padding : JPadding.values()) {
  27. System.out.println("-----------------------------------------");
  28. System.out.println(algorithm + "/" + workingMode + "/" + padding);
  29. byte[] encrypt = JTripleDES.encrypt(str.getBytes(), key, algorithm.getName(), workingMode.getName(),
  30. padding.getName());
  31. System.out.println("加密:" + encoder.encodeToString(encrypt, "UTF-8"));
  32. System.out.println("解密:" + new String(
  33. JTripleDES.decrypt(encrypt, key, algorithm.getName(), workingMode.getName(), padding.getName())));
  34. System.out.print("输出流加密:" + file.getAbsolutePath());
  35. OutputStream out = JTripleDES.wrap(new FileOutputStream(file), key, algorithm.getName(),
  36. workingMode.getName(), padding.getName());
  37. out.write(str.getBytes());
  38. out.flush();
  39. out.close();
  40. System.out.println();
  41. System.out.print("输入流解密:");
  42. InputStream in = JTripleDES.wrap(new FileInputStream(file), key, algorithm.getName(),
  43. workingMode.getName(), padding.getName());
  44. byte[] buffer = new byte[1024];
  45. int len = in.read(buffer);
  46. System.out.println(new String(buffer, 0, len));
  47. }
  48. }
  49. }
  50. }
  51. }

测试结果
原串:jianggujin111111
算法:DES
密钥:STgOq5j+W2I=
—————————————–
DES/ECB/NoPadding
加密:xReaksflVhoskcgCSbx32g==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/ECB/PKCS5Padding
加密:xReaksflVhoskcgCSbx32nAMw70j5Q4d
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/CBC/NoPadding
加密:xReaksflVhoAZm1EazC1pA==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/CBC/PKCS5Padding
加密:xReaksflVhoAZm1EazC1pABSl+6raOnY
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/CFB/NoPadding
加密:lv3gnIv2ARWE+EURnN8qBQ==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/CFB/PKCS5Padding
加密:lv3gnIv2ARWE+EURnN8qBeyGlXXo79Vj
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/OFB/NoPadding
加密:lv3gnIv2ARWdOf4IFFr2RA==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/OFB/PKCS5Padding
加密:lv3gnIv2ARWdOf4IFFr2REpbraYN8vK1
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/CTR/NoPadding
加密:lv3gnIv2ARX96lBo58xf8A==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DES/CTR/PKCS5Padding
加密:lv3gnIv2ARX96lBo58xf8A==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
算法:DESede
密钥:+MQqI5g+c2GJbdOnTFRUkQ2zdgj96hX4
—————————————–
DESede/ECB/NoPadding
加密:4DfUS0dkn0Yf1NIThNPzRA==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/ECB/PKCS5Padding
加密:4DfUS0dkn0Yf1NIThNPzRG7eKaTs2NC9
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/CBC/NoPadding
加密:4DfUS0dkn0Y5mctZQQs4lg==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/CBC/PKCS5Padding
加密:4DfUS0dkn0Y5mctZQQs4liPGg8/jDzsk
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/CFB/NoPadding
加密:2huLYOt44ucXsUXX2fQuqw==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/CFB/PKCS5Padding
加密:2huLYOt44ucXsUXX2fQuq85yWpJ1qxaq
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/OFB/NoPadding
加密:2huLYOt44uedB3fPCBGGmA==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/OFB/PKCS5Padding
加密:2huLYOt44uedB3fPCBGGmBmF7G+qCWDe
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/CTR/NoPadding
加密:2huLYOt44ucNRy7fPNJGmA==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111
—————————————–
DESede/CTR/PKCS5Padding
加密:2huLYOt44ucNRy7fPNJGmA==
解密:jianggujin111111
输出流加密:F:\workspace\java\eclipse\JCodec\TripleDESTest.dat
输入流解密:jianggujin111111

相关文章