DES&3DES算法原理及C#和JS实现的方法是什么

前端开发   发布日期:2023年09月08日   浏览次数:452

这篇文章主要介绍“DES&3DES算法原理及C#和JS实现的方法是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“DES&3DES算法原理及C#和JS实现的方法是什么”文章能帮助大家解决问题。

一、简介

1、DES 简介

DES 全称为 Data Encryption Standard,即数据加密标准,是一种使用密钥加密的块算法,1977 年被美国联邦政府的国家标准局确定为联邦资料处理标准(FIPS),并授权在非密级政府通信中使用,随后该算法在国际上广泛流传开来。

在很长时间内,许多人心目中“密码生成”与 DES 一直是个同义词。直到 1997 年 NIST(美国国家标准与技术研究院)开始公开征集更安全的加密算法以替代 DES,并在 2001 年推出了更加安全的 AES(Advanced Encryption Standard)高级加密标准。

优点:

  • Feistel 网络的轮数可以任意增加;

  • 解密与轮函数 f 无关,轮函数f也不需要有逆函数;

  • 轮函数可以设计得足够复杂;

  • 加密和解密可以使用完全相同的结构来实现。

缺点:

  • 分组比较短;

  • 密钥太短;

  • 密码生命周期短;

  • 运算速度较慢。

2、3DES 简介

其实并不是直接由 DES 过渡到 AES,还有一个 3DES 统治时期。3DES 也称 Triple DES,它使用 3 条 56 位的密钥对数据进行三次加密。

3DES 算法通过对 DES 算法进行改进,增加 DES 的密钥长度来避免类似的攻击,针对每个数据块进行三次 DES 加密;因此,3DES 加密算法并非什么新的加密算法,是 DES 的一个更安全的变形,它以 DES 为基本模块,通过组合分组方法设计出分组加密算法。

相比 DES,3DES 因密钥长度变长,安全性有所提高,但其处理速度不高。因此又出现了 AES 加密算法,AES 较于 3DES 速度更快、安全性也更高。

加密:

为了兼容普通的 DES,3DES 并没有直接使用 加密->加密->加密 的方式,而是采用了 加密->解密->加密 的方式。

当三重密钥均相同时,前两步相互抵消,相当于仅实现了一次加密,因此可实现对普通 DES 加密算法的兼容。

解密:

3DES 解密过程,与加密过程相反,即逆序使用密钥。是以密钥 3、密钥 2、密钥 1的顺序执行 解密->加密->解密。

二、C# 代码实现

1、DES

  1. // 测试(密钥需要是八位字符)
  2. string jiamihou = DesEncrypt("TestString", "11111222", false); // 57fe567eaa866373f851a526f07d9e26
  3. string jiamiqian = DesDecrypt(jiamihou32, "11111222");
  4. /// <summary>
  5. /// DES加密字符串
  6. /// </summary>
  7. /// <param name="deseninstr">待加密的字符串</param>
  8. /// <param name="deskey">加密密钥,要求为8位</param>
  9. /// <param name="isupper">返回大写密文,false:小写</param>
  10. /// <returns>加密成功返回加密后的字符串,失败返回源串</returns>
  11. public static string DesEncrypt(string deseninstr, string deskey, bool isupper = true)
  12. {
  13. StringBuilder stringBuilder = new StringBuilder();
  14. try
  15. {
  16. DESCryptoServiceProvider des = new DESCryptoServiceProvider();
  17. byte[] inputByteArray = Encoding.UTF8.GetBytes(deseninstr);
  18. des.Key = Encoding.UTF8.GetBytes(deskey);
  19. des.IV = Encoding.UTF8.GetBytes(deskey); // 当 mode 为 CBC 时,偏移量必传
  20. des.Mode=CipherMode.ECB; // 为空默认 CBC
  21. MemoryStream memoryStream = new MemoryStream();
  22. CryptoStream cryptoStream = new CryptoStream(memoryStream, des.CreateEncryptor(), CryptoStreamMode.Write);
  23. cryptoStream.Write(inputByteArray, 0, inputByteArray.Length);
  24. cryptoStream.FlushFinalBlock();
  25. foreach (byte bb in memoryStream.ToArray())
  26. {
  27. stringBuilder.AppendFormat(isupper ? "{0:X2}" : "{0:x2}", bb);
  28. }
  29. return stringBuilder.ToString();
  30. }
  31. catch (Exception ex)
  32. {
  33. return deseninstr;
  34. }
  35. }
  36. /// <summary>
  37. /// DES解密字符串
  38. /// </summary>
  39. /// <param name="desdeinstr">待解密的字符串</param>
  40. /// <param name="deskey">解密密钥,要求为8位</param>
  41. /// <returns>解密成功返回解密后的字符串,失败返源串</returns>
  42. public static string DesDecrypt(string desdeinstr, string deskey)
  43. {
  44. MemoryStream memoryStream = new MemoryStream();
  45. try
  46. {
  47. DESCryptoServiceProvider des = new DESCryptoServiceProvider();
  48. byte[] inputByteArray = new byte[desdeinstr.Length / 2];
  49. for (int ii = 0; ii < desdeinstr.Length / 2; ii++)
  50. {
  51. int intt = (Convert.ToInt32(desdeinstr.Substring(ii * 2, 2), 16));
  52. inputByteArray[ii] = (byte)intt;
  53. }
  54. des.Key = Encoding.UTF8.GetBytes(deskey);
  55. des.IV = Encoding.UTF8.GetBytes(deskey); // 当 mode 为 CBC 时,偏移量必传
  56. des.Mode = CipherMode.ECB; // 为空默认 CBC
  57. CryptoStream cs = new CryptoStream(memoryStream, des.CreateDecryptor(), CryptoStreamMode.Write);
  58. cs.Write(inputByteArray, 0, inputByteArray.Length);
  59. cs.FlushFinalBlock();
  60. return Encoding.UTF8.GetString(memoryStream.ToArray());
  61. }
  62. catch
  63. {
  64. return desdeinstr;
  65. }
  66. }

2、3DES

密文采用 Base64 格式输出。

疑问解答:三次加解密操作会运用三个不同的 Key,但是我们只传入了一个密钥,怎么回事?

3DES 密钥必须为 24 位,为 DES 的 3 倍,经测试得出结论:

TripleDESCryptoServiceProvider 内部将密钥分成 3 份,进行了加密解密三重操作。

我们把 24 位字符串分成三部分,如果三部分均相等,或前两部分相等,就会报错:"Specified key is a known weak key for 'TripleDES' and cannot be used."--指定的密钥是'TripleDES'的已知弱密钥,不能使用。

  1. // 测试
  2. string jiamihou16 = SecurityDES.Des3Encrypt("TestString", "111112222233333444445555", "12345678"); // yJGf3qgWyoAQeaPY2S5Etg==
  3. string jiamihou32 = SecurityDES.Des3Decrypt(jiamihou16, "111112222233333444445555", "12345678");
  4. /// <summary>
  5. /// 3DES 加密
  6. /// </summary>
  7. /// <param name="des3eninstr"></param>
  8. /// <param name="des3key">24 位</param>
  9. /// <param name="des3iv">8 位</param>
  10. /// <returns></returns>
  11. public static string Des3Encrypt(string des3eninstr, string des3key, string des3iv)
  12. {
  13. string encryptPassword = string.Empty;
  14. SymmetricAlgorithm algorithm = new TripleDESCryptoServiceProvider();
  15. algorithm.Key = Encoding.UTF8.GetBytes(des3key);// Convert.FromBase64String(des3key);
  16. algorithm.IV = Encoding.UTF8.GetBytes(des3iv);
  17. algorithm.Mode = CipherMode.ECB;
  18. algorithm.Padding = PaddingMode.PKCS7;
  19. ICryptoTransform transform = algorithm.CreateEncryptor();
  20. byte[] data = Encoding.UTF8.GetBytes(des3eninstr);
  21. MemoryStream memoryStream = new MemoryStream();
  22. CryptoStream cryptoStream = new CryptoStream(memoryStream, transform, CryptoStreamMode.Write);
  23. cryptoStream.Write(data, 0, data.Length);
  24. cryptoStream.FlushFinalBlock();
  25. encryptPassword = Convert.ToBase64String(memoryStream.ToArray());
  26. memoryStream.Close();
  27. cryptoStream.Close();
  28. return encryptPassword;
  29. }
  30. /// <summary>
  31. /// 3DES 解密
  32. /// </summary>
  33. /// <param name="des3deinstr">密文 Base64</param>
  34. /// <param name="des3key">24 位</param>
  35. /// <param name="des3iv">8 位</param>
  36. /// <returns></returns>
  37. public static string Des3Decrypt(string des3deinstr, string des3key, string des3iv)
  38. {
  39. string decryptPassword = string.Empty;
  40. SymmetricAlgorithm algorithm = new TripleDESCryptoServiceProvider();
  41. algorithm.Key = Encoding.UTF8.GetBytes(des3key);
  42. algorithm.IV = Encoding.UTF8.GetBytes(des3iv);
  43. algorithm.Mode = CipherMode.ECB;
  44. algorithm.Padding = PaddingMode.PKCS7;
  45. ICryptoTransform transform = algorithm.CreateDecryptor(algorithm.Key, algorithm.IV);
  46. byte[] buffer = Convert.FromBase64String(des3deinstr);
  47. MemoryStream memoryStream = new MemoryStream(buffer);
  48. CryptoStream cryptoStream = new CryptoStream(memoryStream, transform, CryptoStreamMode.Read);
  49. StreamReader reader = new StreamReader(cryptoStream, System.Text.Encoding.ASCII);
  50. decryptPassword = reader.ReadToEnd();
  51. reader.Close();
  52. cryptoStream.Close();
  53. memoryStream.Close();
  54. return decryptPassword;
  55. }

三、js 语言实现

以下是通过 crypto-js.js 实现。

1、DES

注意:mode 为空默认 CBC,此时偏移量 iv 不可为空。

注意:密钥可用位数为 8,如果超过 8 位以后的对加密结果无影响,且不会报错。

  1. // 先引入 js 文件
  2. <script src="http://cdn.bootcdn.net/ajax/libs/crypto-js/4.0.0/crypto-js.js"></script>
  3. // npm(Node.js package manager)方式
  4. > npm install crypto-js
  5. // 调用方法 message() 查看测试结果
  6. function message(){
  7. var outdata_value = encryptByDES("TestString", "11111222");
  8. alert(outdata_value) // 57fe567eaa866373f851a526f07d9e26
  9. console.log("outdata_value-aes_encrypt:", outdata_value);
  10. outdata_value = decryptByDES(outdata_value, "11111222");
  11. alert(outdata_value)
  12. console.log("outdata_value-aes_decrypt:", outdata_value);
  13. }
  14. //DES 加密
  15. function encryptByDES(deseninstr, keystr, ivstr = keystr) {
  16. var keybyte = CryptoJS.enc.Utf8.parse(keystr);
  17. var ivbyte = CryptoJS.enc.Utf8.parse(ivstr);
  18. let afterEncrypt = CryptoJS.DES.encrypt(deseninstr, keybyte, {
  19. iv: ivbyte, // 当 mode 为 CBC 时,偏移量必传
  20. mode: CryptoJS.mode.ECB, // 为空默认 CBC
  21. padding: CryptoJS.pad.Pkcs7
  22. }).ciphertext.toString()
  23. console.log(afterEncrypt)
  24. return afterEncrypt
  25. }
  26. //DES 解密
  27. function decryptByDES(desdeinstr, keystr, ivstr = keystr) {
  28. var keybyte = CryptoJS.enc.Utf8.parse(keystr);
  29. var ivbyte = CryptoJS.enc.Utf8.parse(ivstr);
  30. var decrypted = CryptoJS.DES.decrypt(
  31. { ciphertext: CryptoJS.enc.Hex.parse(desdeinstr) },
  32. keybyte,
  33. {
  34. iv: ivbyte, // 当 mode 为 CBC 时,偏移量必传
  35. mode: CryptoJS.mode.ECB, // 为空默认 CBC
  36. padding: CryptoJS.pad.Pkcs7
  37. }
  38. );
  39. console.log(decrypted);
  40. var result_value = decrypted.toString(CryptoJS.enc.Utf8);
  41. return result_value;
  42. }

2、3DES

  1. // 调用方法 message() 查看测试结果
  2. function message() {
  3. var outdata_value = encryptByDES("TestString", "111112222233333444445555");
  4. alert(outdata_value) // yJGf3qgWyoAQeaPY2S5Etg==
  5. console.log("outdata_value-3des_encrypt:", outdata_value);
  6. outdata_value = decryptByDES(outdata_value, "111112222233333444445555");
  7. alert(outdata_value)
  8. console.log("outdata_value-3des_decrypt:", outdata_value);
  9. }
  10. // 加密 密钥需为 24 位,偏移量需为 8 位
  11. function encryptByDES(deseninstr, keystr) {
  12. var keybyte = CryptoJS.enc.Utf8.parse(keystr);
  13. //var ivbyte = CryptoJS.enc.Utf8.parse(ivstr);
  14. var encrypted = CryptoJS.TripleDES.encrypt(deseninstr, keybyte, {
  15. // iv: ivbyte, // 当 mode 为 CBC 时,偏移量必传
  16. mode: CryptoJS.mode.ECB,
  17. padding: CryptoJS.pad.Pkcs7
  18. });
  19. return encrypted.toString();
  20. }
  21. // 解密 密钥需为 24 位,偏移量需为 8 位
  22. function decryptByDES(desdeinstr, keystr) {
  23. var keybyte = CryptoJS.enc.Utf8.parse(keystr);
  24. //var ivbyte = CryptoJS.enc.Utf8.parse(ivstr);
  25. var decrypted = CryptoJS.TripleDES.decrypt(desdeinstr, keybyte, {
  26. // iv: ivbyte, // 当 mode 为 CBC 时,偏移量必传
  27. mode: CryptoJS.mode.ECB,
  28. padding: CryptoJS.pad.Pkcs7
  29. });
  30. return decrypted.toString(CryptoJS.enc.Utf8);
  31. }

以上就是DES&amp;3DES算法原理及C#和JS实现的方法是什么的详细内容,更多关于DES&amp;3DES算法原理及C#和JS实现的方法是什么的资料请关注九品源码其它相关文章!