Encrypt.cs 53 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380
  1. using System;
  2. using System.IO;
  3. using System.Linq;
  4. using System.Security.Cryptography;
  5. using System.Text;
  6. using System.Text.RegularExpressions;
  7. namespace Masuit.Tools.Security
  8. {
  9. /// <summary>
  10. /// 常用的加密解密算法
  11. /// </summary>
  12. public static class Encrypt
  13. {
  14. #region DES对称加密解密
  15. /// <summary>
  16. /// 加密密钥,默认取“masuit”的MD5值
  17. /// </summary>
  18. public static string DefaultEncryptKey = "masuit".MDString2();
  19. /// <summary>
  20. /// 使用默认加密
  21. /// </summary>
  22. /// <param name="strText">被加密的字符串</param>
  23. /// <returns>加密后的数据</returns>
  24. public static string DesEncrypt(this string strText)
  25. {
  26. try
  27. {
  28. return DesEncrypt(strText, DefaultEncryptKey);
  29. }
  30. catch
  31. {
  32. return "";
  33. }
  34. }
  35. /// <summary>
  36. /// 使用默认解密
  37. /// </summary>
  38. /// <param name="strText">需要解密的 字符串</param>
  39. /// <returns>解密后的数据</returns>
  40. public static string DesDecrypt(this string strText)
  41. {
  42. try
  43. {
  44. return DesDecrypt(strText, DefaultEncryptKey);
  45. }
  46. catch
  47. {
  48. return "";
  49. }
  50. }
  51. /// <summary>
  52. /// 加密字符串
  53. /// 加密密钥必须为8位
  54. /// </summary>
  55. /// <param name="strText">被加密的字符串</param>
  56. /// <param name="strEncrKey">8位长度密钥</param>
  57. /// <returns>加密后的数据</returns>
  58. public static string DesEncrypt(this string strText, string strEncrKey)
  59. {
  60. if (strEncrKey.Length < 8)
  61. {
  62. throw new Exception("密钥长度无效,密钥必须是8位!");
  63. }
  64. StringBuilder ret = new StringBuilder();
  65. using var des = new DESCryptoServiceProvider();
  66. byte[] inputByteArray = Encoding.Default.GetBytes(strText);
  67. des.Key = Encoding.ASCII.GetBytes(strEncrKey.Substring(0, 8));
  68. des.IV = Encoding.ASCII.GetBytes(strEncrKey.Substring(0, 8));
  69. MemoryStream ms = new MemoryStream();
  70. using var cs = new CryptoStream(ms, des.CreateEncryptor(), CryptoStreamMode.Write);
  71. cs.Write(inputByteArray, 0, inputByteArray.Length);
  72. cs.FlushFinalBlock();
  73. foreach (byte b in ms.ToArray())
  74. {
  75. ret.AppendFormat($"{b:X2}");
  76. }
  77. return ret.ToString();
  78. }
  79. /// <summary>
  80. /// 加密字符串
  81. /// 加密密钥必须为8位
  82. /// </summary>
  83. /// <param name="strText">被加密的字符串</param>
  84. /// <param name="desKey"></param>
  85. /// <param name="desIV"></param>
  86. /// <returns>加密后的数据</returns>
  87. public static string DesEncrypt(this string strText, byte[] desKey, byte[] desIV)
  88. {
  89. StringBuilder ret = new StringBuilder();
  90. using var des = new DESCryptoServiceProvider();
  91. byte[] inputByteArray = Encoding.Default.GetBytes(strText);
  92. des.Key = desKey;
  93. des.IV = desIV;
  94. MemoryStream ms = new MemoryStream();
  95. using var cs = new CryptoStream(ms, des.CreateEncryptor(), CryptoStreamMode.Write);
  96. cs.Write(inputByteArray, 0, inputByteArray.Length);
  97. cs.FlushFinalBlock();
  98. foreach (byte b in ms.ToArray())
  99. {
  100. ret.AppendFormat($"{b:X2}");
  101. }
  102. return ret.ToString();
  103. }
  104. /// <summary>
  105. /// DES加密文件
  106. /// </summary>
  107. /// <param name="fin">文件输入流</param>
  108. /// <param name="outFilePath">文件输出路径</param>
  109. /// <param name="strEncrKey">加密密钥</param>
  110. public static void DesEncrypt(this FileStream fin, string outFilePath, string strEncrKey)
  111. {
  112. byte[] iv =
  113. {
  114. 0x12,
  115. 0x34,
  116. 0x56,
  117. 0x78,
  118. 0x90,
  119. 0xAB,
  120. 0xCD,
  121. 0xEF
  122. };
  123. var byKey = Encoding.UTF8.GetBytes(strEncrKey.Substring(0, 8));
  124. using var fout = new FileStream(outFilePath, FileMode.OpenOrCreate, FileAccess.Write);
  125. fout.SetLength(0);
  126. byte[] bin = new byte[100];
  127. long rdlen = 0;
  128. long totlen = fin.Length;
  129. DES des = new DESCryptoServiceProvider();
  130. var encStream = new CryptoStream(fout, des.CreateEncryptor(byKey, iv), CryptoStreamMode.Write);
  131. while (rdlen < totlen)
  132. {
  133. var len = fin.Read(bin, 0, 100);
  134. encStream.Write(bin, 0, len);
  135. rdlen += len;
  136. }
  137. }
  138. /// <summary>
  139. /// DES加密文件
  140. /// </summary>
  141. /// <param name="fin">文件输入流</param>
  142. /// <param name="outFilePath">文件输出路径</param>
  143. /// <param name="desKey"></param>
  144. /// <param name="desIV"></param>
  145. public static void DesEncrypt(this FileStream fin, string outFilePath, byte[] desKey, byte[] desIV)
  146. {
  147. using var fout = new FileStream(outFilePath, FileMode.OpenOrCreate, FileAccess.Write);
  148. fout.SetLength(0);
  149. byte[] bin = new byte[100];
  150. long rdlen = 0;
  151. long totlen = fin.Length;
  152. DES des = new DESCryptoServiceProvider();
  153. var encStream = new CryptoStream(fout, des.CreateEncryptor(desKey, desIV), CryptoStreamMode.Write);
  154. while (rdlen < totlen)
  155. {
  156. var len = fin.Read(bin, 0, 100);
  157. encStream.Write(bin, 0, len);
  158. rdlen += len;
  159. }
  160. }
  161. /// <summary>
  162. /// DES解密文件
  163. /// </summary>
  164. /// <param name="fin">输入文件流</param>
  165. /// <param name="outFilePath">文件输出路径</param>
  166. /// <param name="sDecrKey">解密密钥</param>
  167. public static void DesDecrypt(this FileStream fin, string outFilePath, string sDecrKey)
  168. {
  169. byte[] iv =
  170. {
  171. 0x12,
  172. 0x34,
  173. 0x56,
  174. 0x78,
  175. 0x90,
  176. 0xAB,
  177. 0xCD,
  178. 0xEF
  179. };
  180. var byKey = Encoding.UTF8.GetBytes(sDecrKey.Substring(0, 8));
  181. using var fout = new FileStream(outFilePath, FileMode.OpenOrCreate, FileAccess.Write);
  182. fout.SetLength(0);
  183. byte[] bin = new byte[100];
  184. long rdlen = 0;
  185. long totlen = fin.Length;
  186. using DES des = new DESCryptoServiceProvider();
  187. var encStream = new CryptoStream(fout, des.CreateDecryptor(byKey, iv), CryptoStreamMode.Write);
  188. while (rdlen < totlen)
  189. {
  190. var len = fin.Read(bin, 0, 100);
  191. encStream.Write(bin, 0, len);
  192. rdlen += len;
  193. }
  194. }
  195. /// <summary>
  196. /// DES解密文件
  197. /// </summary>
  198. /// <param name="fin">输入文件流</param>
  199. /// <param name="outFilePath">文件输出路径</param>
  200. /// <param name="desKey"></param>
  201. /// <param name="desIV"></param>
  202. public static void DesDecrypt(this FileStream fin, string outFilePath, byte[] desKey, byte[] desIV)
  203. {
  204. using var fout = new FileStream(outFilePath, FileMode.OpenOrCreate, FileAccess.Write);
  205. fout.SetLength(0);
  206. byte[] bin = new byte[100];
  207. long rdlen = 0;
  208. long totlen = fin.Length;
  209. using DES des = new DESCryptoServiceProvider();
  210. var encStream = new CryptoStream(fout, des.CreateDecryptor(desKey, desIV), CryptoStreamMode.Write);
  211. while (rdlen < totlen)
  212. {
  213. var len = fin.Read(bin, 0, 100);
  214. encStream.Write(bin, 0, len);
  215. rdlen += len;
  216. }
  217. }
  218. /// <summary>
  219. /// DES解密算法
  220. /// 密钥为8位
  221. /// </summary>
  222. /// <param name="pToDecrypt">需要解密的字符串</param>
  223. /// <param name="sKey">密钥</param>
  224. /// <returns>解密后的数据</returns>
  225. public static string DesDecrypt(this string pToDecrypt, string sKey)
  226. {
  227. if (sKey.Length < 8)
  228. {
  229. throw new Exception("密钥长度无效,密钥必须是8位!");
  230. }
  231. var ms = new MemoryStream();
  232. using var des = new DESCryptoServiceProvider();
  233. var inputByteArray = new byte[pToDecrypt.Length / 2];
  234. for (int x = 0; x < pToDecrypt.Length / 2; x++)
  235. {
  236. int i = Convert.ToInt32(pToDecrypt.Substring(x * 2, 2), 16);
  237. inputByteArray[x] = (byte)i;
  238. }
  239. des.Key = Encoding.ASCII.GetBytes(sKey.Substring(0, 8));
  240. des.IV = Encoding.ASCII.GetBytes(sKey.Substring(0, 8));
  241. using var cs = new CryptoStream(ms, des.CreateDecryptor(), CryptoStreamMode.Write);
  242. cs.Write(inputByteArray, 0, inputByteArray.Length);
  243. cs.FlushFinalBlock();
  244. return Encoding.Default.GetString(ms.ToArray());
  245. }
  246. /// <summary>
  247. /// DES解密算法
  248. /// 密钥为8位
  249. /// </summary>
  250. /// <param name="pToDecrypt">需要解密的字符串</param>
  251. /// <param name="desKey"></param>
  252. /// <param name="desIV"></param>
  253. /// <returns>解密后的数据</returns>
  254. public static string DesDecrypt(this string pToDecrypt, byte[] desKey, byte[] desIV)
  255. {
  256. var ms = new MemoryStream();
  257. using var des = new DESCryptoServiceProvider();
  258. var inputByteArray = new byte[pToDecrypt.Length / 2];
  259. for (int x = 0; x < pToDecrypt.Length / 2; x++)
  260. {
  261. int i = Convert.ToInt32(pToDecrypt.Substring(x * 2, 2), 16);
  262. inputByteArray[x] = (byte)i;
  263. }
  264. des.Key = desKey;
  265. des.IV = desIV;
  266. using var cs = new CryptoStream(ms, des.CreateDecryptor(), CryptoStreamMode.Write);
  267. cs.Write(inputByteArray, 0, inputByteArray.Length);
  268. cs.FlushFinalBlock();
  269. return Encoding.Default.GetString(ms.ToArray());
  270. }
  271. #endregion
  272. #region 对称加密算法AES RijndaelManaged加密解密
  273. private static readonly string Default_AES_Key = "@#kim123";
  274. private static byte[] Keys =
  275. {
  276. 0x41,
  277. 0x72,
  278. 0x65,
  279. 0x79,
  280. 0x6F,
  281. 0x75,
  282. 0x6D,
  283. 0x79,
  284. 0x53,
  285. 0x6E,
  286. 0x6F,
  287. 0x77,
  288. 0x6D,
  289. 0x61,
  290. 0x6E,
  291. 0x3F
  292. };
  293. /// <summary>
  294. /// 生成符合AES加密规则的密钥
  295. /// </summary>
  296. /// <param name="length"></param>
  297. /// <returns></returns>
  298. public static string GenerateAesKey(int length)
  299. {
  300. var crypto = new AesCryptoServiceProvider
  301. {
  302. KeySize = length,
  303. BlockSize = 128
  304. };
  305. crypto.GenerateKey();
  306. return Convert.ToBase64String(crypto.Key);
  307. }
  308. /// <summary>
  309. /// 对称加密算法AES RijndaelManaged加密(RijndaelManaged(AES)算法是块式加密算法)
  310. /// </summary>
  311. /// <param name="encryptString">待加密字符串</param>
  312. /// <param name="mode">加密模式</param>
  313. /// <returns>加密结果字符串</returns>
  314. public static string AESEncrypt(this string encryptString, CipherMode mode = CipherMode.CBC)
  315. {
  316. return AESEncrypt(encryptString, Default_AES_Key, mode);
  317. }
  318. /// <summary>
  319. /// 对称加密算法AES RijndaelManaged加密(RijndaelManaged(AES)算法是块式加密算法)
  320. /// </summary>
  321. /// <param name="encryptString">待加密字符串</param>
  322. /// <param name="encryptKey">加密密钥,须半角字符</param>
  323. /// <param name="mode">加密模式</param>
  324. /// <returns>加密结果字符串</returns>
  325. public static string AESEncrypt(this string encryptString, string encryptKey, CipherMode mode = CipherMode.CBC)
  326. {
  327. encryptKey = GetSubString(encryptKey, 32, "");
  328. encryptKey = encryptKey.PadRight(32, ' ');
  329. using var rijndaelProvider = new RijndaelManaged
  330. {
  331. Key = Encoding.UTF8.GetBytes(encryptKey.Substring(0, 32)),
  332. IV = Keys,
  333. Mode = mode
  334. };
  335. using ICryptoTransform rijndaelEncrypt = rijndaelProvider.CreateEncryptor();
  336. byte[] inputData = Encoding.UTF8.GetBytes(encryptString);
  337. byte[] encryptedData = rijndaelEncrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  338. return Convert.ToBase64String(encryptedData);
  339. }
  340. /// <summary>
  341. /// 对称加密算法AES RijndaelManaged加密(RijndaelManaged(AES)算法是块式加密算法)
  342. /// </summary>
  343. /// <param name="encryptString">待加密字符串</param>
  344. /// <param name="options">加密选项</param>
  345. /// <returns>加密结果字符串</returns>
  346. public static string AESEncrypt(this string encryptString, RijndaelManaged options)
  347. {
  348. using ICryptoTransform rijndaelEncrypt = options.CreateEncryptor();
  349. byte[] inputData = Encoding.UTF8.GetBytes(encryptString);
  350. byte[] encryptedData = rijndaelEncrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  351. return Convert.ToBase64String(encryptedData);
  352. }
  353. /// <summary>
  354. /// 对称加密算法AES RijndaelManaged加密(RijndaelManaged(AES)算法是块式加密算法)
  355. /// </summary>
  356. /// <param name="encryptString">待加密字符串</param>
  357. /// <param name="encryptKey">加密密钥,须半角字符</param>
  358. /// <param name="mode">加密模式</param>
  359. /// <returns>加密结果字符串</returns>
  360. public static string AESEncrypt(this string encryptString, byte[] encryptKey, CipherMode mode = CipherMode.CBC)
  361. {
  362. using var rijndaelProvider = new RijndaelManaged
  363. {
  364. Key = encryptKey,
  365. IV = Keys,
  366. Mode = mode
  367. };
  368. using ICryptoTransform rijndaelEncrypt = rijndaelProvider.CreateEncryptor();
  369. byte[] inputData = Encoding.UTF8.GetBytes(encryptString);
  370. byte[] encryptedData = rijndaelEncrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  371. return Convert.ToBase64String(encryptedData);
  372. }
  373. /// <summary>
  374. /// 对称加密算法AES RijndaelManaged解密字符串
  375. /// </summary>
  376. /// <param name="decryptString">待解密的字符串</param>
  377. /// <param name="mode">加密模式</param>
  378. /// <returns>解密成功返回解密后的字符串,失败返源串</returns>
  379. public static string AESDecrypt(this string decryptString, CipherMode mode = CipherMode.CBC)
  380. {
  381. return AESDecrypt(decryptString, Default_AES_Key, mode);
  382. }
  383. /// <summary>
  384. /// 对称加密算法AES RijndaelManaged解密字符串
  385. /// </summary>
  386. /// <param name="decryptString">待解密的字符串</param>
  387. /// <param name="decryptKey">解密密钥,和加密密钥相同</param>
  388. /// <param name="mode">加密模式</param>
  389. /// <returns>解密成功返回解密后的字符串,失败返回空</returns>
  390. public static string AESDecrypt(this string decryptString, string decryptKey, CipherMode mode = CipherMode.CBC)
  391. {
  392. try
  393. {
  394. decryptKey = GetSubString(decryptKey, 32, "");
  395. decryptKey = decryptKey.PadRight(32, ' ');
  396. using var rijndaelProvider = new RijndaelManaged()
  397. {
  398. Key = Encoding.UTF8.GetBytes(decryptKey),
  399. IV = Keys,
  400. Mode = mode
  401. };
  402. using ICryptoTransform rijndaelDecrypt = rijndaelProvider.CreateDecryptor();
  403. byte[] inputData = Convert.FromBase64String(decryptString);
  404. byte[] decryptedData = rijndaelDecrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  405. return Encoding.UTF8.GetString(decryptedData);
  406. }
  407. catch
  408. {
  409. return string.Empty;
  410. }
  411. }
  412. /// <summary>
  413. /// 对称加密算法AES RijndaelManaged解密字符串
  414. /// </summary>
  415. /// <param name="decryptString">待解密的字符串</param>
  416. /// <param name="options">加密选项</param>
  417. /// <returns>解密成功返回解密后的字符串,失败返回空</returns>
  418. public static string AESDecrypt(this string decryptString, RijndaelManaged options)
  419. {
  420. try
  421. {
  422. using ICryptoTransform rijndaelDecrypt = options.CreateDecryptor();
  423. byte[] inputData = Convert.FromBase64String(decryptString);
  424. byte[] decryptedData = rijndaelDecrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  425. return Encoding.UTF8.GetString(decryptedData);
  426. }
  427. catch
  428. {
  429. return string.Empty;
  430. }
  431. }
  432. /// <summary>
  433. /// 对称加密算法AES RijndaelManaged解密字符串
  434. /// </summary>
  435. /// <param name="decryptString">待解密的字符串</param>
  436. /// <param name="decryptKey">解密密钥,和加密密钥相同</param>
  437. /// <param name="mode">加密模式</param>
  438. /// <returns>解密成功返回解密后的字符串,失败返回空</returns>
  439. public static string AESDecrypt(this string decryptString, byte[] decryptKey, CipherMode mode = CipherMode.CBC)
  440. {
  441. try
  442. {
  443. using var rijndaelProvider = new RijndaelManaged()
  444. {
  445. Key = decryptKey,
  446. IV = Keys,
  447. Mode = mode
  448. };
  449. using ICryptoTransform rijndaelDecrypt = rijndaelProvider.CreateDecryptor();
  450. byte[] inputData = Convert.FromBase64String(decryptString);
  451. byte[] decryptedData = rijndaelDecrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  452. return Encoding.UTF8.GetString(decryptedData);
  453. }
  454. catch
  455. {
  456. return string.Empty;
  457. }
  458. }
  459. /// <summary>
  460. /// 按字节长度(按字节,一个汉字为2个字节)取得某字符串的一部分
  461. /// </summary>
  462. /// <param name="sourceString">源字符串</param>
  463. /// <param name="length">所取字符串字节长度</param>
  464. /// <param name="tailString">附加字符串(当字符串不够长时,尾部所添加的字符串,一般为"...")</param>
  465. /// <returns>某字符串的一部分</returns>
  466. private static string GetSubString(this string sourceString, int length, string tailString)
  467. {
  468. return GetSubString(sourceString, 0, length, tailString);
  469. }
  470. /// <summary>
  471. /// 按字节长度(按字节,一个汉字为2个字节)取得某字符串的一部分
  472. /// </summary>
  473. /// <param name="sourceString">源字符串</param>
  474. /// <param name="startIndex">索引位置,以0开始</param>
  475. /// <param name="length">所取字符串字节长度</param>
  476. /// <param name="tailString">附加字符串(当字符串不够长时,尾部所添加的字符串,一般为"...")</param>
  477. /// <returns>某字符串的一部分</returns>
  478. private static string GetSubString(this string sourceString, int startIndex, int length, string tailString)
  479. {
  480. //当是日文或韩文时(注:中文的范围:\u4e00 - \u9fa5, 日文在\u0800 - \u4e00, 韩文为\xAC00-\xD7A3)
  481. if (Regex.IsMatch(sourceString, "[\u0800-\u4e00]+") || Regex.IsMatch(sourceString, "[\xAC00-\xD7A3]+"))
  482. {
  483. //当截取的起始位置超出字段串长度时
  484. if (startIndex >= sourceString.Length)
  485. {
  486. return string.Empty;
  487. }
  488. return sourceString.Substring(startIndex, length + startIndex > sourceString.Length ? (sourceString.Length - startIndex) : length);
  489. }
  490. //中文字符,如"中国人民abcd123"
  491. if (length <= 0)
  492. {
  493. return string.Empty;
  494. }
  495. byte[] bytesSource = Encoding.Default.GetBytes(sourceString);
  496. //当字符串长度大于起始位置
  497. if (bytesSource.Length > startIndex)
  498. {
  499. int endIndex = bytesSource.Length;
  500. //当要截取的长度在字符串的有效长度范围内
  501. if (bytesSource.Length > (startIndex + length))
  502. {
  503. endIndex = length + startIndex;
  504. }
  505. else
  506. {
  507. //当不在有效范围内时,只取到字符串的结尾
  508. length = bytesSource.Length - startIndex;
  509. tailString = "";
  510. }
  511. var anResultFlag = new int[length];
  512. int nFlag = 0;
  513. //字节大于127为双字节字符
  514. for (int i = startIndex; i < endIndex; i++)
  515. {
  516. if (bytesSource[i] > 127)
  517. {
  518. nFlag++;
  519. if (nFlag == 3)
  520. {
  521. nFlag = 1;
  522. }
  523. }
  524. else
  525. {
  526. nFlag = 0;
  527. }
  528. anResultFlag[i] = nFlag;
  529. }
  530. //最后一个字节为双字节字符的一半
  531. if ((bytesSource[endIndex - 1] > 127) && (anResultFlag[length - 1] == 1))
  532. {
  533. length++;
  534. }
  535. byte[] bsResult = new byte[length];
  536. Array.Copy(bytesSource, startIndex, bsResult, 0, length);
  537. var myResult = Encoding.Default.GetString(bsResult);
  538. myResult += tailString;
  539. return myResult;
  540. }
  541. return string.Empty;
  542. }
  543. /// <summary>
  544. /// 加密文件流
  545. /// </summary>
  546. /// <param name="fs">需要加密的文件流</param>
  547. /// <param name="decryptKey">加密密钥</param>
  548. /// <param name="mode">加密模式</param>
  549. /// <returns>加密流</returns>
  550. public static CryptoStream AESEncryptStrream(this FileStream fs, string decryptKey, CipherMode mode = CipherMode.CBC)
  551. {
  552. decryptKey = GetSubString(decryptKey, 32, "");
  553. decryptKey = decryptKey.PadRight(32, ' ');
  554. using var rijndaelProvider = new RijndaelManaged()
  555. {
  556. Key = Encoding.UTF8.GetBytes(decryptKey),
  557. IV = Keys,
  558. Mode = mode
  559. };
  560. using var encrypto = rijndaelProvider.CreateEncryptor();
  561. return new CryptoStream(fs, encrypto, CryptoStreamMode.Write);
  562. }
  563. /// <summary>
  564. /// 加密文件流
  565. /// </summary>
  566. /// <param name="fs">需要加密的文件流</param>
  567. /// <param name="decryptKey">加密密钥</param>
  568. /// <param name="mode">加密模式</param>
  569. /// <returns>加密流</returns>
  570. public static CryptoStream AESEncryptStrream(this FileStream fs, byte[] decryptKey, CipherMode mode = CipherMode.CBC)
  571. {
  572. using var rijndaelProvider = new RijndaelManaged()
  573. {
  574. Key = decryptKey,
  575. IV = Keys,
  576. Mode = mode
  577. };
  578. using var encrypto = rijndaelProvider.CreateEncryptor();
  579. return new CryptoStream(fs, encrypto, CryptoStreamMode.Write);
  580. }
  581. /// <summary>
  582. /// 解密文件流
  583. /// </summary>
  584. /// <param name="fs">需要解密的文件流</param>
  585. /// <param name="decryptKey">解密密钥</param>
  586. /// <param name="mode">加密模式</param>
  587. /// <returns>加密流</returns>
  588. public static CryptoStream AESDecryptStream(this FileStream fs, string decryptKey, CipherMode mode = CipherMode.CBC)
  589. {
  590. decryptKey = GetSubString(decryptKey, 32, "");
  591. decryptKey = decryptKey.PadRight(32, ' ');
  592. using var rijndaelProvider = new RijndaelManaged()
  593. {
  594. Key = Encoding.UTF8.GetBytes(decryptKey),
  595. IV = Keys,
  596. Mode = mode
  597. };
  598. using var decrypto = rijndaelProvider.CreateDecryptor();
  599. return new CryptoStream(fs, decrypto, CryptoStreamMode.Read);
  600. }
  601. /// <summary>
  602. /// 解密文件流
  603. /// </summary>
  604. /// <param name="fs">需要解密的文件流</param>
  605. /// <param name="decryptKey">解密密钥</param>
  606. /// <param name="mode">加密模式</param>
  607. /// <returns>加密流</returns>
  608. public static CryptoStream AESDecryptStream(this FileStream fs, byte[] decryptKey, CipherMode mode = CipherMode.CBC)
  609. {
  610. using var rijndaelProvider = new RijndaelManaged()
  611. {
  612. Key = decryptKey,
  613. IV = Keys,
  614. Mode = mode
  615. };
  616. using var decrypto = rijndaelProvider.CreateDecryptor();
  617. return new CryptoStream(fs, decrypto, CryptoStreamMode.Read);
  618. }
  619. /// <summary>
  620. /// 对指定文件AES加密
  621. /// </summary>
  622. /// <param name="input">源文件流</param>
  623. /// <param name="mode">加密模式</param>
  624. /// <param name="outputPath">输出文件路径</param>
  625. public static void AESEncryptFile(this FileStream input, string outputPath, CipherMode mode = CipherMode.CBC)
  626. {
  627. using var fren = new FileStream(outputPath, FileMode.Create);
  628. using var enfr = AESEncryptStrream(fren, Default_AES_Key, mode);
  629. byte[] bytearrayinput = new byte[input.Length];
  630. input.Read(bytearrayinput, 0, bytearrayinput.Length);
  631. enfr.Write(bytearrayinput, 0, bytearrayinput.Length);
  632. }
  633. /// <summary>
  634. /// 对指定的文件AES解密
  635. /// </summary>
  636. /// <param name="input">源文件流</param>
  637. /// <param name="mode">加密模式</param>
  638. /// <param name="outputPath">输出文件路径</param>
  639. public static void AESDecryptFile(this FileStream input, string outputPath, CipherMode mode = CipherMode.CBC)
  640. {
  641. using FileStream frde = new FileStream(outputPath, FileMode.Create);
  642. using CryptoStream defr = AESDecryptStream(input, Default_AES_Key, mode);
  643. byte[] bytearrayoutput = new byte[1024];
  644. while (true)
  645. {
  646. var count = defr.Read(bytearrayoutput, 0, bytearrayoutput.Length);
  647. frde.Write(bytearrayoutput, 0, count);
  648. if (count < bytearrayoutput.Length)
  649. {
  650. break;
  651. }
  652. }
  653. }
  654. #endregion
  655. #region Base64加密解密
  656. /// <summary>
  657. /// Base64加密
  658. /// </summary>
  659. /// <param name="str">需要加密的字符串</param>
  660. /// <returns>加密后的数据</returns>
  661. public static string Base64Encrypt(this string str)
  662. {
  663. byte[] encbuff = Encoding.UTF8.GetBytes(str);
  664. return Convert.ToBase64String(encbuff);
  665. }
  666. /// <summary>
  667. /// Base64解密
  668. /// </summary>
  669. /// <param name="str">需要解密的字符串</param>
  670. /// <returns>解密后的数据</returns>
  671. public static string Base64Decrypt(this string str)
  672. {
  673. try
  674. {
  675. byte[] decbuff = Convert.FromBase64String(str);
  676. return Encoding.UTF8.GetString(decbuff);
  677. }
  678. catch
  679. {
  680. return str;
  681. }
  682. }
  683. #endregion
  684. /// <summary>
  685. /// SHA256函数
  686. /// </summary>
  687. /// <param name="str">原始字符串</param>
  688. /// <returns>SHA256结果(返回长度为44字节的字符串)</returns>
  689. public static string SHA256(this string str)
  690. {
  691. byte[] sha256Data = Encoding.UTF8.GetBytes(str);
  692. using var sha256 = new SHA256Managed();
  693. byte[] result = sha256.ComputeHash(sha256Data);
  694. return Convert.ToBase64String(result); //返回长度为44字节的字符串
  695. }
  696. #region MD5摘要算法
  697. #region 对字符串进行MD5摘要
  698. /// <summary>
  699. /// 对字符串进行MD5摘要
  700. /// </summary>
  701. /// <param name="message">需要摘要的字符串</param>
  702. /// <returns>MD5摘要字符串</returns>
  703. public static string MDString(this string message)
  704. {
  705. MD5 md5 = MD5.Create();
  706. byte[] buffer = Encoding.Default.GetBytes(message);
  707. byte[] bytes = md5.ComputeHash(buffer);
  708. return GetHexString(bytes);
  709. }
  710. /// <summary>
  711. /// 对字符串进行MD5二次摘要
  712. /// </summary>
  713. /// <param name="message">需要摘要的字符串</param>
  714. /// <returns>MD5摘要字符串</returns>
  715. public static string MDString2(this string message) => MDString(MDString(message));
  716. /// <summary>
  717. /// MD5 三次摘要算法
  718. /// </summary>
  719. /// <param name="s">需要摘要的字符串</param>
  720. /// <returns>MD5摘要字符串</returns>
  721. public static string MDString3(this string s)
  722. {
  723. using MD5 md5 = MD5.Create();
  724. byte[] bytes = Encoding.ASCII.GetBytes(s);
  725. byte[] bytes1 = md5.ComputeHash(bytes);
  726. byte[] bytes2 = md5.ComputeHash(bytes1);
  727. byte[] bytes3 = md5.ComputeHash(bytes2);
  728. return GetHexString(bytes3);
  729. }
  730. /// <summary>
  731. /// 对字符串进行MD5加盐摘要
  732. /// </summary>
  733. /// <param name="message">需要摘要的字符串</param>
  734. /// <param name="salt">盐</param>
  735. /// <returns>MD5摘要字符串</returns>
  736. public static string MDString(this string message, string salt) => MDString(message + salt);
  737. /// <summary>
  738. /// 对字符串进行MD5二次加盐摘要
  739. /// </summary>
  740. /// <param name="message">需要摘要的字符串</param>
  741. /// <param name="salt">盐</param>
  742. /// <returns>MD5摘要字符串</returns>
  743. public static string MDString2(this string message, string salt) => MDString(MDString(message + salt), salt);
  744. /// <summary>
  745. /// MD5 三次摘要算法
  746. /// </summary>
  747. /// <param name="s">需要摘要的字符串</param>
  748. /// <param name="salt">盐</param>
  749. /// <returns>MD5摘要字符串</returns>
  750. public static string MDString3(this string s, string salt)
  751. {
  752. using MD5 md5 = MD5.Create();
  753. byte[] bytes = Encoding.ASCII.GetBytes(s + salt);
  754. byte[] bytes1 = md5.ComputeHash(bytes);
  755. byte[] bytes2 = md5.ComputeHash(bytes1);
  756. byte[] bytes3 = md5.ComputeHash(bytes2);
  757. return GetHexString(bytes3);
  758. }
  759. #endregion
  760. #region 获取文件的MD5值
  761. /// <summary>
  762. /// 获取文件的MD5值
  763. /// </summary>
  764. /// <param name="fileName">需要求MD5值的文件的文件名及路径</param>
  765. /// <returns>MD5摘要字符串</returns>
  766. public static string MDFile(this string fileName)
  767. {
  768. using var fs = new BufferedStream(File.Open(fileName, FileMode.Open, FileAccess.Read), 1048576);
  769. using MD5 md5 = MD5.Create();
  770. byte[] bytes = md5.ComputeHash(fs);
  771. return GetHexString(bytes);
  772. }
  773. /// <summary>
  774. /// 计算文件的sha256
  775. /// </summary>
  776. /// <param name="stream"></param>
  777. /// <returns></returns>
  778. public static string SHA256(this Stream stream)
  779. {
  780. using var fs = new BufferedStream(stream, 1048576);
  781. SHA256Managed sha = new SHA256Managed();
  782. byte[] checksum = sha.ComputeHash(fs);
  783. return BitConverter.ToString(checksum).Replace("-", string.Empty);
  784. }
  785. /// <summary>
  786. /// 获取数据流的MD5摘要值
  787. /// </summary>
  788. /// <param name="stream"></param>
  789. /// <returns>MD5摘要字符串</returns>
  790. public static string MDString(this Stream stream)
  791. {
  792. using var fs = new BufferedStream(stream, 1048576);
  793. using MD5 md5 = MD5.Create();
  794. byte[] bytes = md5.ComputeHash(fs);
  795. var mdstr = GetHexString(bytes);
  796. stream.Position = 0;
  797. return mdstr;
  798. }
  799. public static string GetHexString(byte[] bytes)
  800. {
  801. var hexArray = new char[bytes.Length << 1];
  802. for (var i = 0; i < hexArray.Length; i += 2)
  803. {
  804. var b = bytes[i >> 1];
  805. hexArray[i] = GetHexValue(b >> 4); // b / 16
  806. hexArray[i + 1] = GetHexValue(b & 0xF); // b % 16
  807. }
  808. return new string(hexArray, 0, hexArray.Length);
  809. static char GetHexValue(int i)
  810. {
  811. if (i < 10)
  812. {
  813. return (char)(i + '0');
  814. }
  815. return (char)(i - 10 + 'a');
  816. }
  817. }
  818. #endregion
  819. #endregion MD5摘要算法
  820. }
  821. /// <summary>
  822. /// RC2加密解密算法
  823. /// </summary>
  824. public static class RC2
  825. {
  826. private static ASCIIEncoding _asciiEncoding;
  827. private static byte[] _iv;
  828. private static byte[] _key;
  829. private static RC2CryptoServiceProvider _rc2Csp;
  830. private static UnicodeEncoding _textConverter;
  831. static RC2()
  832. {
  833. InitializeComponent();
  834. }
  835. private static void InitializeComponent()
  836. {
  837. _key = new byte[]
  838. {
  839. 106,
  840. 51,
  841. 25,
  842. 141,
  843. 157,
  844. 142,
  845. 23,
  846. 111,
  847. 234,
  848. 159,
  849. 187,
  850. 154,
  851. 215,
  852. 34,
  853. 37,
  854. 204
  855. };
  856. _iv = new byte[]
  857. {
  858. 135,
  859. 186,
  860. 133,
  861. 136,
  862. 184,
  863. 149,
  864. 153,
  865. 144
  866. };
  867. _asciiEncoding = new ASCIIEncoding();
  868. _textConverter = new UnicodeEncoding();
  869. _rc2Csp = new RC2CryptoServiceProvider();
  870. }
  871. #region 将文本数据加密后写入一个文件
  872. /// <summary>
  873. /// 将文本数据加密后写入一个文件,其中,这个文件是用InitBinFile建立的,这个文件将被分成十块,
  874. /// 用来分别保存10组不同的数据,第一个byte位保留,第2位到第21位分别用来存放每块数据的长度,但
  875. /// 一个byte的取值为0-127,所以,用两个byte来存放一个长度。
  876. /// </summary>
  877. /// <param name="toEncryptText">要加密的文本数据</param>
  878. /// <param name="filePath">要写入的文件</param>
  879. /// <param name="dataIndex">写入第几块,取值为1--10</param>
  880. /// <returns>是否操作成功</returns>
  881. public static bool EncryptToFile(this string toEncryptText, string filePath, int dataIndex)
  882. {
  883. var r = false;
  884. if ((dataIndex > 10) && (dataIndex < 1))
  885. {
  886. return r;
  887. }
  888. //打开要写入的文件,主要是为了保持原文件的内容不丢失
  889. var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  890. var index = new byte[10261];
  891. //将读取的内容写到byte数组
  892. tmpFileStream.Read(index, 0, 10261);
  893. tmpFileStream.Close();
  894. //定义基本的加密转换运算
  895. using var Encryptor = _rc2Csp.CreateEncryptor(_key, _iv);
  896. var msEncrypt = new MemoryStream();
  897. //在此加密转换流中,加密将从csEncrypt,加密后,结果在msEncrypt流中。
  898. using var csEncrypt = new CryptoStream(msEncrypt, Encryptor, CryptoStreamMode.Write);
  899. //将要加密的文本转换成UTF-16 编码,保存在tmp数组。
  900. var tmp = _textConverter.GetBytes(toEncryptText);
  901. //将tmp输入csEncrypt,将通过Encryptor来加密。
  902. csEncrypt.Write(tmp, 0, tmp.Length);
  903. //输出到msEnctypt
  904. csEncrypt.FlushFinalBlock();
  905. //将流转成byte[]
  906. var encrypted = msEncrypt.ToArray();
  907. if (encrypted.Length > 1024)
  908. return false;
  909. //得到加密后数据的大小,将结果存在指定的位置。
  910. index[dataIndex * 2 - 1] = Convert.ToByte(Convert.ToString(encrypted.Length / 128));
  911. index[dataIndex * 2] = Convert.ToByte(Convert.ToString(encrypted.Length % 128));
  912. //将加密后的结果写入index(覆盖)
  913. for (var i = 0; i < encrypted.Length; i++)
  914. index[1024 * (dataIndex - 1) + 21 + i] = encrypted[i];
  915. //建立文件流
  916. tmpFileStream = new FileStream(filePath, FileMode.Truncate, FileAccess.Write, FileShare.None, 1024, true);
  917. //写文件
  918. tmpFileStream.Write(index, 0, 10261);
  919. tmpFileStream.Flush();
  920. r = true;
  921. tmpFileStream.Close();
  922. return r;
  923. }
  924. #endregion
  925. #region 从一个文件中解密出一段文本,其中,这个文件是由InitBinFile建立的,并且由 EncryptToFile加密的
  926. /// <summary>
  927. /// 从一个文件中解密出一段文本,其中,这个文件是由InitBinFile建立的,并且由 EncryptToFile加密的
  928. /// </summary>
  929. /// <param name="filePath">要解密的文件</param>
  930. /// <param name="dataIndex">要从哪一个块中解密</param>
  931. /// <returns>解密后的文本</returns>
  932. public static string DecryptFromFile(this string filePath, int dataIndex)
  933. {
  934. if (dataIndex > 10 && dataIndex < 1)
  935. {
  936. return "";
  937. }
  938. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  939. using var decryptor = _rc2Csp.CreateDecryptor(_key, _iv);
  940. var msDecrypt = new MemoryStream();
  941. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  942. var index = new byte[10261];
  943. tmpFileStream.Read(index, 0, 10261);
  944. var count = index[dataIndex * 2 - 1] * 128 + index[dataIndex * 2];
  945. var tmp = new byte[count];
  946. Array.Copy(index, 1024 * (dataIndex - 1) + 21, tmp, 0, count);
  947. csDecrypt.Write(tmp, 0, count);
  948. csDecrypt.FlushFinalBlock();
  949. var decrypted = msDecrypt.ToArray();
  950. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  951. }
  952. #endregion
  953. #region 将一段文本加密后保存到一个文件
  954. /// <summary>
  955. /// 将一段文本加密后保存到一个文件
  956. /// </summary>
  957. /// <param name="toEncryptText">要加密的文本数据</param>
  958. /// <param name="filePath">要保存的文件</param>
  959. /// <returns>是否加密成功</returns>
  960. public static void EncryptToFile(this string toEncryptText, string filePath)
  961. {
  962. using var tmpFileStream = new FileStream(filePath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.None, 1024, true);
  963. using var encryptor = _rc2Csp.CreateEncryptor(_key, _iv);
  964. var msEncrypt = new MemoryStream();
  965. using var csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write);
  966. var tmp = _textConverter.GetBytes(toEncryptText);
  967. csEncrypt.Write(tmp, 0, tmp.Length);
  968. csEncrypt.FlushFinalBlock();
  969. var encrypted = msEncrypt.ToArray();
  970. tmpFileStream.Write(encrypted, 0, encrypted.Length);
  971. }
  972. #endregion
  973. #region 将一个被加密的文件解密
  974. /// <summary>
  975. /// 将一个被加密的文件解密
  976. /// </summary>
  977. /// <param name="filePath">要解密的文件</param>
  978. /// <returns>解密后的文本</returns>
  979. public static string DecryptFromFile(this string filePath)
  980. {
  981. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  982. using var decryptor = _rc2Csp.CreateDecryptor(_key, _iv);
  983. var msDecrypt = new MemoryStream();
  984. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  985. var tmp = new byte[tmpFileStream.Length];
  986. tmpFileStream.Read(tmp, 0, tmp.Length);
  987. csDecrypt.Write(tmp, 0, tmp.Length);
  988. csDecrypt.FlushFinalBlock();
  989. var decrypted = msDecrypt.ToArray();
  990. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  991. }
  992. #endregion
  993. #region 将文本数据加密后写入一个文件
  994. /// <summary>
  995. /// 将文本数据加密后写入一个文件,其中,这个文件是用InitBinFile建立的,这个文件将被分成十块,
  996. /// 用来分别保存10组不同的数据,第一个byte位保留,第2位到第21位分别用来存放每块数据的长度,但
  997. /// 一个byte的取值为0-127,所以,用两个byte来存放一个长度。
  998. /// </summary>
  999. /// <param name="toEncryptText">要加密的文本数据</param>
  1000. /// <param name="filePath">要写入的文件</param>
  1001. /// <param name="dataIndex">写入第几块,取值为1--10</param>
  1002. /// <param name="IV">初始化向量</param>
  1003. /// <param name="Key">加密密匙</param>
  1004. /// <returns>是否操作成功</returns>
  1005. public static void EncryptToFile(this string toEncryptText, string filePath, int dataIndex, byte[] IV, byte[] Key)
  1006. {
  1007. if ((dataIndex > 10) && (dataIndex < 1))
  1008. {
  1009. return;
  1010. }
  1011. //打开要写入的文件,主要是为了保持原文件的内容不丢失
  1012. using var fs = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  1013. var index = new byte[10261];
  1014. //将读取的内容写到byte数组
  1015. fs.Read(index, 0, 10261);
  1016. //定义基本的加密转换运算
  1017. using var encryptor = _rc2Csp.CreateEncryptor(Key, IV);
  1018. var msEncrypt = new MemoryStream();
  1019. //在此加密转换流中,加密将从csEncrypt,加密后,结果在msEncrypt流中。
  1020. using var csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write);
  1021. var tmp = _textConverter.GetBytes(toEncryptText);
  1022. //将tmp输入csEncrypt,将通过Encryptor来加密。
  1023. csEncrypt.Write(tmp, 0, tmp.Length);
  1024. //输出到msEnctypt
  1025. csEncrypt.FlushFinalBlock();
  1026. //将流转成byte[]
  1027. var encrypted = msEncrypt.ToArray();
  1028. if (encrypted.Length > 1024)
  1029. {
  1030. return;
  1031. }
  1032. //得到加密后数据的大小,将结果存在指定的位置。
  1033. index[dataIndex * 2 - 1] = Convert.ToByte(Convert.ToString(encrypted.Length / 128));
  1034. index[dataIndex * 2] = Convert.ToByte(Convert.ToString(encrypted.Length % 128));
  1035. //将加密后的结果写入index(覆盖)
  1036. for (int i = 0; i < encrypted.Length; i++)
  1037. {
  1038. index[1024 * (dataIndex - 1) + 21 + i] = encrypted[i];
  1039. }
  1040. //建立文件流
  1041. using var newStream = new FileStream(filePath, FileMode.Truncate, FileAccess.Write, FileShare.None, 1024, true);
  1042. newStream.Write(index, 0, 10261);
  1043. newStream.Flush();
  1044. }
  1045. #endregion
  1046. #region 从一个文件中解密出一段文本
  1047. /// <summary>
  1048. /// 从一个文件中解密出一段文本,其中,这个文件是由InitBinFile建立的,并且由 EncryptToFile加密的
  1049. /// </summary>
  1050. /// <param name="filePath">要解密的文件</param>
  1051. /// <param name="dataIndex">要从哪一个块中解密</param>
  1052. /// <param name="iv">初始化向量</param>
  1053. /// <param name="key">解密密匙</param>
  1054. /// <returns>解密后的文本</returns>
  1055. public static string DecryptFromFile(this string filePath, int dataIndex, byte[] iv, byte[] key)
  1056. {
  1057. if ((dataIndex > 10) && (dataIndex < 1))
  1058. {
  1059. return "";
  1060. }
  1061. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  1062. using var decryptor = _rc2Csp.CreateDecryptor(key, iv);
  1063. var msDecrypt = new MemoryStream();
  1064. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  1065. var index = new byte[10261];
  1066. tmpFileStream.Read(index, 0, 10261);
  1067. var count = index[dataIndex * 2 - 1] * 128 + index[dataIndex * 2];
  1068. var tmp = new byte[count];
  1069. Array.Copy(index, 1024 * (dataIndex - 1) + 21, tmp, 0, count);
  1070. csDecrypt.Write(tmp, 0, count);
  1071. csDecrypt.FlushFinalBlock();
  1072. var decrypted = msDecrypt.ToArray();
  1073. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  1074. }
  1075. #endregion
  1076. #region 将一段文本加密后保存到一个文件
  1077. /// <summary>
  1078. /// 将一段文本加密后保存到一个文件
  1079. /// </summary>
  1080. /// <param name="toEncryptText">要加密的文本数据</param>
  1081. /// <param name="filePath">要保存的文件</param>
  1082. /// <param name="iv">初始化向量</param>
  1083. /// <param name="key">加密密匙</param>
  1084. /// <returns>是否加密成功</returns>
  1085. public static void EncryptToFile(this string toEncryptText, string filePath, byte[] iv, byte[] key)
  1086. {
  1087. using var tmpFileStream = new FileStream(filePath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.None, 1024, true);
  1088. using var encryptor = _rc2Csp.CreateEncryptor(key, iv);
  1089. var msEncrypt = new MemoryStream();
  1090. using var csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write);
  1091. var tmp = _textConverter.GetBytes(toEncryptText);
  1092. csEncrypt.Write(tmp, 0, tmp.Length);
  1093. csEncrypt.FlushFinalBlock();
  1094. var encrypted = msEncrypt.ToArray();
  1095. tmpFileStream.Write(encrypted, 0, encrypted.Length);
  1096. tmpFileStream.Flush();
  1097. }
  1098. #endregion
  1099. #region 将一个被加密的文件解密
  1100. /// <summary>
  1101. /// 将一个被加密的文件解密
  1102. /// </summary>
  1103. /// <param name="filePath">要解密的文件</param>
  1104. /// <param name="iv">初始化向量</param>
  1105. /// <param name="key">解密密匙</param>
  1106. /// <returns>解密后的文本</returns>
  1107. public static string DecryptFromFile(this string filePath, byte[] iv, byte[] key)
  1108. {
  1109. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  1110. using var decryptor = _rc2Csp.CreateDecryptor(key, iv);
  1111. var msDecrypt = new MemoryStream();
  1112. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  1113. var tmp = new byte[tmpFileStream.Length];
  1114. tmpFileStream.Read(tmp, 0, tmp.Length);
  1115. csDecrypt.Write(tmp, 0, tmp.Length);
  1116. csDecrypt.FlushFinalBlock();
  1117. var decrypted = msDecrypt.ToArray();
  1118. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  1119. }
  1120. #endregion
  1121. #region 设置加密或解密的初始化向量
  1122. /// <summary>
  1123. /// 设置加密或解密的初始化向量
  1124. /// </summary>
  1125. /// <param name="s">长度等于8的ASCII字符集的字符串</param>
  1126. public static void SetIV(this string s)
  1127. {
  1128. if (s.Length != 8)
  1129. {
  1130. _iv = null;
  1131. return;
  1132. }
  1133. try
  1134. {
  1135. _iv = _asciiEncoding.GetBytes(s);
  1136. }
  1137. catch (Exception)
  1138. {
  1139. _iv = null;
  1140. }
  1141. }
  1142. #endregion
  1143. #region 设置加密或解密的密匙
  1144. /// <summary>
  1145. /// 设置加密或解密的密匙
  1146. /// </summary>
  1147. /// <param name="s">长度等于16的ASCII字符集的字符串</param>
  1148. public static void SetKey(this string s)
  1149. {
  1150. if (s.Length != 16)
  1151. {
  1152. _key = null;
  1153. return;
  1154. }
  1155. try
  1156. {
  1157. _key = _asciiEncoding.GetBytes(s);
  1158. }
  1159. catch (Exception)
  1160. {
  1161. _key = null;
  1162. }
  1163. }
  1164. #endregion
  1165. }
  1166. /// <summary>
  1167. /// 对称加密解密算法类
  1168. /// </summary>
  1169. public static class Rijndael
  1170. {
  1171. private static string _key;
  1172. private static SymmetricAlgorithm _mobjCryptoService;
  1173. /// <summary>
  1174. /// 对称加密类的构造函数
  1175. /// </summary>
  1176. public static void SymmetricMethod()
  1177. {
  1178. _mobjCryptoService = new RijndaelManaged();
  1179. _key = "Guz(%&hj7x89H$yuBI0456FtmaT5&fvHUFCy76*h%(HilJ$lhj!y6&(*jkP87jH7";
  1180. }
  1181. /// <summary>
  1182. /// 获得密钥
  1183. /// </summary>
  1184. /// <returns>密钥</returns>
  1185. private static byte[] GetLegalKey()
  1186. {
  1187. var sTemp = _key;
  1188. _mobjCryptoService.GenerateKey();
  1189. var bytTemp = _mobjCryptoService.Key;
  1190. var keyLength = bytTemp.Length;
  1191. if (sTemp.Length > keyLength)
  1192. {
  1193. sTemp = sTemp.Substring(0, keyLength);
  1194. }
  1195. else if (sTemp.Length < keyLength)
  1196. {
  1197. sTemp = sTemp.PadRight(keyLength, ' ');
  1198. }
  1199. return Encoding.ASCII.GetBytes(sTemp);
  1200. }
  1201. /// <summary>
  1202. /// 获得初始向量IV
  1203. /// </summary>
  1204. /// <returns>初试向量IV</returns>
  1205. private static byte[] GetLegalIV()
  1206. {
  1207. var sTemp = "E4ghj*Ghg7!rNIfb&95GUY86GfghUber57HBh(u%g6HJ($jhWk7&!hg4ui%$hjk";
  1208. _mobjCryptoService.GenerateIV();
  1209. var bytTemp = _mobjCryptoService.IV;
  1210. var ivLength = bytTemp.Length;
  1211. if (sTemp.Length > ivLength)
  1212. {
  1213. sTemp = sTemp.Substring(0, ivLength);
  1214. }
  1215. else if (sTemp.Length < ivLength)
  1216. {
  1217. sTemp = sTemp.PadRight(ivLength, ' ');
  1218. }
  1219. return Encoding.ASCII.GetBytes(sTemp);
  1220. }
  1221. /// <summary>
  1222. /// 加密方法
  1223. /// </summary>
  1224. /// <param name="source">待加密的串</param>
  1225. /// <returns>经过加密的串</returns>
  1226. public static string Encrypto(this string source)
  1227. {
  1228. var bytIn = Encoding.UTF8.GetBytes(source);
  1229. var ms = new MemoryStream();
  1230. _mobjCryptoService.Key = GetLegalKey();
  1231. _mobjCryptoService.IV = GetLegalIV();
  1232. using var encrypto = _mobjCryptoService.CreateEncryptor();
  1233. using var cs = new CryptoStream(ms, encrypto, CryptoStreamMode.Write);
  1234. cs.Write(bytIn, 0, bytIn.Length);
  1235. cs.FlushFinalBlock();
  1236. var bytOut = ms.ToArray();
  1237. return Convert.ToBase64String(bytOut);
  1238. }
  1239. /// <summary>
  1240. /// 解密方法
  1241. /// </summary>
  1242. /// <param name="source">待解密的串</param>
  1243. /// <returns>经过解密的串</returns>
  1244. public static string Decrypto(this string source)
  1245. {
  1246. var bytIn = Convert.FromBase64String(source);
  1247. var ms = new MemoryStream(bytIn, 0, bytIn.Length);
  1248. _mobjCryptoService.Key = GetLegalKey();
  1249. _mobjCryptoService.IV = GetLegalIV();
  1250. using var encrypto = _mobjCryptoService.CreateDecryptor();
  1251. using var cs = new CryptoStream(ms, encrypto, CryptoStreamMode.Read);
  1252. using var sr = new StreamReader(cs);
  1253. return sr.ReadToEnd();
  1254. }
  1255. }
  1256. }