您好,登錄后才能下訂單哦!
這篇文章主要講解了“.NET中常見(jiàn)的加解密算法有哪些”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“.NET中常見(jiàn)的加解密算法有哪些”吧!
不可逆加密是指將原文加密成密文以后,無(wú)法將密文解密成原文。
MD5的算法是公開(kāi)的,無(wú)論是哪種語(yǔ)言,只要需要加密的字符串是相同的,那么經(jīng)過(guò)MD5加密以后生成的結(jié)果都是一樣的。
.NET框架中已經(jīng)幫我們實(shí)現(xiàn)好了MD5加密,請(qǐng)看下面的例子:
using System; using System.Collections.Generic; using System.Linq; using System.Security.Cryptography; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { public class MD5Encrypt { #region MD5 /// <summary> /// MD5加密,和動(dòng)網(wǎng)上的16/32位MD5加密結(jié)果相同, /// 使用的UTF8編碼 /// </summary> /// <param name="source">待加密字串</param> /// <param name="length">16或32值之一,其它則采用.net默認(rèn)MD5加密算法</param> /// <returns>加密后的字串</returns> public static string Encrypt(string source, int length = 32)//默認(rèn)參數(shù) { if (string.IsNullOrEmpty(source)) return string.Empty; HashAlgorithm provider = CryptoConfig.CreateFromName("MD5") as HashAlgorithm; byte[] bytes = Encoding.UTF8.GetBytes(source);//這里需要區(qū)別編碼的 byte[] hashValue = provider.ComputeHash(bytes); StringBuilder sb = new StringBuilder(); switch (length) { case 16://16位密文是32位密文的9到24位字符 for (int i = 4; i < 12; i++) { sb.Append(hashValue[i].ToString("x2")); } break; case 32: for (int i = 0; i < 16; i++) { sb.Append(hashValue[i].ToString("x2")); } break; default: for (int i = 0; i < hashValue.Length; i++) { sb.Append(hashValue[i].ToString("x2")); } break; } return sb.ToString(); } #endregion MD5 } }
Main()方法調(diào)用:
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { class Program { static void Main(string[] args) { // MD5 Console.WriteLine(MD5Encrypt.Encrypt("1")); Console.WriteLine(MD5Encrypt.Encrypt("1")); Console.WriteLine(MD5Encrypt.Encrypt("123456孫悟空")); Console.WriteLine(MD5Encrypt.Encrypt("113456孫悟空")); Console.WriteLine(MD5Encrypt.Encrypt("113456孫悟空113456孫悟空113456孫悟空113456孫悟空113456孫悟空113456孫悟空113456孫悟空")); Console.ReadKey(); } } }
結(jié)果:
應(yīng)用:
從上面的例子中可以看出,只要字符串相同,那么加密以后的結(jié)果就是一樣的,利用MD5的這個(gè)特性,可以用來(lái)做密碼校驗(yàn)。在注冊(cè)的時(shí)候把密碼用MD5加密然后保存到數(shù)據(jù)庫(kù)里面,數(shù)據(jù)庫(kù)里面保存的是密文,別人無(wú)法看到。登錄的時(shí)候,在把密碼經(jīng)過(guò)MD5加密,然后用加密后的密文和數(shù)據(jù)庫(kù)里面保存的密文進(jìn)行比對(duì),如果相同,則證明密碼是一樣的;如果不同,證明密碼是錯(cuò)誤的。
注意:MD5是不能解密的,網(wǎng)上的解密都是基于撞庫(kù)原理的:即將原文和密文保存到數(shù)據(jù)庫(kù)中,每次利用密文去和數(shù)據(jù)庫(kù)里保存的密文進(jìn)行比對(duì),如果比對(duì)成功,則解密了。為了防止撞庫(kù),可以使密碼復(fù)雜一些,例如加鹽:即在密碼的后面加上一段后綴然后加密后在保存到數(shù)據(jù)庫(kù)。登錄的時(shí)候,在密碼后面加上同樣的后綴,然后加密以后和數(shù)據(jù)庫(kù)保存的密碼進(jìn)行比對(duì)。
例如下載VS安裝文件,官網(wǎng)下載的文件才是權(quán)威的,但是有時(shí)會(huì)去系統(tǒng)之家這一類(lèi)的網(wǎng)站下載,如何保證在系統(tǒng)之家下載的安裝文件和官網(wǎng)發(fā)布的文件是一樣的呢?這時(shí)就可以利用MD5進(jìn)行判斷。官方在發(fā)布VS安裝文件的同時(shí),也會(huì)發(fā)布一個(gè)根據(jù)該文件生成的MD5碼,在系統(tǒng)之家下載完安裝文件以后,可以對(duì)該安裝文件進(jìn)行一次MD5加密,然后比對(duì)官方發(fā)布的MD5碼和生成的MD5碼,如果相同,則證明下載的文件就是官方方便的。那么如何對(duì)文件進(jìn)行MD5呢?請(qǐng)看下面的例子:
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Security.Cryptography; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { public class MD5Encrypt { #region MD5 /// <summary> /// MD5加密,和動(dòng)網(wǎng)上的16/32位MD5加密結(jié)果相同, /// 使用的UTF8編碼 /// </summary> /// <param name="source">待加密字串</param> /// <param name="length">16或32值之一,其它則采用.net默認(rèn)MD5加密算法</param> /// <returns>加密后的字串</returns> public static string Encrypt(string source, int length = 32)//默認(rèn)參數(shù) { if (string.IsNullOrEmpty(source)) return string.Empty; HashAlgorithm provider = CryptoConfig.CreateFromName("MD5") as HashAlgorithm; byte[] bytes = Encoding.UTF8.GetBytes(source);//這里需要區(qū)別編碼的 byte[] hashValue = provider.ComputeHash(bytes); StringBuilder sb = new StringBuilder(); switch (length) { case 16://16位密文是32位密文的9到24位字符 for (int i = 4; i < 12; i++) { sb.Append(hashValue[i].ToString("x2")); } break; case 32: for (int i = 0; i < 16; i++) { sb.Append(hashValue[i].ToString("x2")); } break; default: for (int i = 0; i < hashValue.Length; i++) { sb.Append(hashValue[i].ToString("x2")); } break; } return sb.ToString(); } #endregion MD5 #region MD5摘要 /// <summary> /// 獲取文件的MD5摘要 /// </summary> /// <param name="fileName"></param> /// <returns></returns> public static string AbstractFile(string fileName) { using (FileStream file = new FileStream(fileName, FileMode.Open)) { return AbstractFile(file); } } /// <summary> /// 根據(jù)stream獲取文件摘要 /// </summary> /// <param name="stream"></param> /// <returns></returns> public static string AbstractFile(Stream stream) { MD5 md5 = new MD5CryptoServiceProvider(); byte[] retVal = md5.ComputeHash(stream); StringBuilder sb = new StringBuilder(); for (int i = 0; i < retVal.Length; i++) { sb.Append(retVal[i].ToString("x2")); } return sb.ToString(); } #endregion } }
Main()方法里面調(diào)用:
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { class Program { static void Main(string[] args) { // MD5 //Console.WriteLine(MD5Encrypt.Encrypt("1")); //Console.WriteLine(MD5Encrypt.Encrypt("1")); //Console.WriteLine(MD5Encrypt.Encrypt("123456孫悟空")); //Console.WriteLine(MD5Encrypt.Encrypt("113456孫悟空")); //Console.WriteLine(MD5Encrypt.Encrypt("113456孫悟空113456孫悟空113456孫悟空113456孫悟空113456孫悟空113456孫悟空113456孫悟空")); // 對(duì)文件進(jìn)行MD5 string md5Abstract1 = MD5Encrypt.AbstractFile(@"E:\EF一對(duì)多.txt"); Console.WriteLine(md5Abstract1); string md5Abstract2 = MD5Encrypt.AbstractFile(@"E:\EF一對(duì)多 - 副本.txt"); Console.WriteLine(md5Abstract2); Console.ReadKey(); } } }
結(jié)果:
可以看出,雖然文件的名稱(chēng)不同,但只要文件的內(nèi)容是相同的,則生成的MD5碼就是相同的。
以百度云為例:假如從百度云上面下載了一個(gè)文件,然后把這個(gè)文件在上傳到百度云就會(huì)急速秒傳。因?yàn)榈谝淮紊蟼鞯臅r(shí)候,百度云會(huì)對(duì)上傳的文件進(jìn)行MD5加密,然后把加密后的MD5碼保存下來(lái)。下載之后再上傳,百度云客戶端會(huì)先對(duì)文件計(jì)算MD5,然后將計(jì)算的MD5和服務(wù)器保存的MD5進(jìn)行對(duì)比,如果一致就不需要在上傳了,只需要把服務(wù)器上文件的名稱(chēng)修改成和上傳文件的名稱(chēng)一致即可。因?yàn)樯蟼鞯奈募诜?wù)器上已經(jīng)存在。(就算修改了文件名稱(chēng),但生成的MD5還是一樣的)
源代碼管理工具實(shí)現(xiàn)判斷文件是否修改,也是根據(jù)MD5進(jìn)行比對(duì)的。
對(duì)稱(chēng)可逆加密:可逆是指加密和解密是可逆的,即可以根據(jù)原文得到密文,也可以根據(jù)密文得到原文。對(duì)稱(chēng)是指加密和解密的密鑰是相同的。下面以DES加密為例。
在示例程序中,密鑰長(zhǎng)度是8位的,寫(xiě)在配置文件中。
讀取配置文件獲取密鑰的代碼如下:
using System; using System.Collections.Generic; using System.Configuration; using System.Linq; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { public static class Constant { public static string DesKey = AppSettings("DesKey", "DesEncript"); private static T AppSettings<T>(string key, T defaultValue) { var v = ConfigurationManager.AppSettings[key]; return String.IsNullOrEmpty(v) ? defaultValue : (T)Convert.ChangeType(v, typeof(T)); } } }
加密和解密的代碼如下:
using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Security.Cryptography; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { /// <summary> /// DES AES Blowfish /// 對(duì)稱(chēng)加密算法的優(yōu)點(diǎn)是速度快, /// 缺點(diǎn)是密鑰管理不方便,要求共享密鑰。 /// 可逆對(duì)稱(chēng)加密 密鑰長(zhǎng)度8 /// </summary> public class DesEncrypt { // 按照8位長(zhǎng)度的密鑰進(jìn)行加密 private static byte[] _rgbKey = ASCIIEncoding.ASCII.GetBytes(Constant.DesKey.Substring(0, 8)); // 對(duì)稱(chēng)算法的初始化向量 private static byte[] _rgbIV = ASCIIEncoding.ASCII.GetBytes(Constant.DesKey.Insert(0, "w").Substring(0, 8)); /// <summary> /// DES 加密 /// </summary> /// <param name="text">需要加密的值</param> /// <returns>加密后的結(jié)果</returns> public static string Encrypt(string text) { DESCryptoServiceProvider dsp = new DESCryptoServiceProvider(); using (MemoryStream memStream = new MemoryStream()) { CryptoStream crypStream = new CryptoStream(memStream, dsp.CreateEncryptor(_rgbKey, _rgbIV), CryptoStreamMode.Write); StreamWriter sWriter = new StreamWriter(crypStream); sWriter.Write(text); sWriter.Flush(); crypStream.FlushFinalBlock(); memStream.Flush(); return Convert.ToBase64String(memStream.GetBuffer(), 0, (int)memStream.Length); } } /// <summary> /// DES解密 /// </summary> /// <param name="encryptText"></param> /// <returns>解密后的結(jié)果</returns> public static string Decrypt(string encryptText) { DESCryptoServiceProvider dsp = new DESCryptoServiceProvider(); byte[] buffer = Convert.FromBase64String(encryptText); using (MemoryStream memStream = new MemoryStream()) { CryptoStream crypStream = new CryptoStream(memStream, dsp.CreateDecryptor(_rgbKey, _rgbIV), CryptoStreamMode.Write); crypStream.Write(buffer, 0, buffer.Length); crypStream.FlushFinalBlock(); return ASCIIEncoding.UTF8.GetString(memStream.ToArray()); } } } }
Main()方法調(diào)用:
string strDes = "張三李四"; string desEn1 = DesEncrypt.Encrypt(strDes); string desDe1 = DesEncrypt.Decrypt(desEn1); Console.WriteLine(strDes.Equals(desDe1));
結(jié)果:
注意:對(duì)稱(chēng)可逆加密的算法是公開(kāi)的。
非對(duì)稱(chēng)可逆加密:可逆是指加密和解密是一樣,即根據(jù)原文可以得到密文,根據(jù)密文也可以得到原文。非對(duì)稱(chēng)是指加密和解密的密鑰是不同的。下面以RSA加密為例:
using System; using System.Collections.Generic; using System.Linq; using System.Security.Cryptography; using System.Text; using System.Threading.Tasks; namespace MyEncriptDemo { /// <summary> /// RSA ECC /// 可逆非對(duì)稱(chēng)加密 /// 非對(duì)稱(chēng)加密算法的優(yōu)點(diǎn)是密鑰管理很方便,缺點(diǎn)是速度慢。 /// </summary> public class RsaEncrypt { /// <summary> /// 獲取加密/解密對(duì) /// 給你一個(gè),是無(wú)法推算出另外一個(gè)的 /// /// Encrypt Decrypt /// </summary> /// <returns>Encrypt Decrypt</returns> public static KeyValuePair<string, string> GetKeyPair() { RSACryptoServiceProvider RSA = new RSACryptoServiceProvider(); string publicKey = RSA.ToXmlString(false); string privateKey = RSA.ToXmlString(true); return new KeyValuePair<string, string>(publicKey, privateKey); } /// <summary> /// 加密:內(nèi)容+加密key /// </summary> /// <param name="content"></param> /// <param name="encryptKey">加密key</param> /// <returns></returns> public static string Encrypt(string content, string encryptKey) { RSACryptoServiceProvider rsa = new RSACryptoServiceProvider(); rsa.FromXmlString(encryptKey); UnicodeEncoding ByteConverter = new UnicodeEncoding(); byte[] DataToEncrypt = ByteConverter.GetBytes(content); byte[] resultBytes = rsa.Encrypt(DataToEncrypt, false); return Convert.ToBase64String(resultBytes); } /// <summary> /// 解密 內(nèi)容+解密key /// </summary> /// <param name="content"></param> /// <param name="decryptKey">解密key</param> /// <returns></returns> public static string Decrypt(string content, string decryptKey) { byte[] dataToDecrypt = Convert.FromBase64String(content); RSACryptoServiceProvider RSA = new RSACryptoServiceProvider(); RSA.FromXmlString(decryptKey); byte[] resultBytes = RSA.Decrypt(dataToDecrypt, false); UnicodeEncoding ByteConverter = new UnicodeEncoding(); return ByteConverter.GetString(resultBytes); } /// <summary> /// 可以合并在一起的,,每次產(chǎn)生一組新的密鑰 /// </summary> /// <param name="content"></param> /// <param name="encryptKey">加密key</param> /// <param name="decryptKey">解密key</param> /// <returns>加密后結(jié)果</returns> private static string Encrypt(string content, out string publicKey, out string privateKey) { RSACryptoServiceProvider rsaProvider = new RSACryptoServiceProvider(); publicKey = rsaProvider.ToXmlString(false); privateKey = rsaProvider.ToXmlString(true); UnicodeEncoding ByteConverter = new UnicodeEncoding(); byte[] DataToEncrypt = ByteConverter.GetBytes(content); byte[] resultBytes = rsaProvider.Encrypt(DataToEncrypt, false); return Convert.ToBase64String(resultBytes); } } }
Main()方法調(diào)用:
// 獲取加密和解密的密鑰 KeyValuePair<string, string> encryptDecrypt = RsaEncrypt.GetKeyPair(); string strValue = "RsaDemo"; string rsaEn1 = RsaEncrypt.Encrypt(strValue, encryptDecrypt.Key);//key是加密的 string rsaDe1 = RsaEncrypt.Decrypt(rsaEn1, encryptDecrypt.Value);//value 解密的 不能反過(guò)來(lái)用的 Console.WriteLine(strValue.Equals(rsaDe1));
結(jié)果:
注意:
1、加密鑰和解密鑰是根據(jù)功能來(lái)劃分的。
2、私鑰和公鑰是根據(jù)鑰匙的公開(kāi)程度來(lái)劃分的,加密鑰可以作為公鑰或者私鑰、解密鑰也可以作為公鑰或者私鑰。
感謝各位的閱讀,以上就是“.NET中常見(jiàn)的加解密算法有哪些”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì).NET中常見(jiàn)的加解密算法有哪些這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。