「Common Encryption and Decryption in C#」では、いくつかの暗号化と復号化の方法が紹介されています。その 1 つが、対称暗号化アルゴリズム DES の使用方法です。今回は、DES のアップグレード版である TripleDES について説明します。
DES と TripleDES の関係については、次のブログ投稿を参照してください。
対称暗号化DESとTripleDES
重要なポイントは、相手が送信した暗号化されたコンテンツを両者が復号できるように、それぞれC#とPHPでTripleDESを使用する方法です。
早速、両端のコードに直接進みましょう。
C# code
<code><span>using</span> System; <span>using</span> System.Security.Cryptography; <span>using</span> System.Text; namespace TripleDes { class Program { <span>static</span><span>void</span> Main(<span>string</span>[] args) { <span>//加密的字符串</span><span>string</span> source = <span>"Happy Father's Day!"</span>; <span>//用来加密的key 但是最终用在des加密的key</span><span>//是这个key的 md5 hash</span><span>string</span> sourceKey = <span>"home"</span>; <span>//获取原始key值的 md5 hash</span><span>byte</span>[] keyBytes = GetKeyMd5Hash(sourceKey); <span>string</span> encryptedStr = DesEncrypt(source, keyBytes); Console.WriteLine(<span>"encrypted string: "</span>+ encryptedStr); <span>string</span> decryptedStr = DesDecrypt(encryptedStr, keyBytes); Console.WriteLine(<span>"Decrypted String: "</span> + decryptedStr); Console.ReadKey(); } <span><span>///</span><span><summary></span></span><span><span>///</span> 获取加密key的 md5 hash,最终DES加密的时候使用这个hash值</span><span><span>///</span><span></summary></span></span><span><span>///</span><span><param name="key"></span>原始key值<span></param></span></span><span><span>///</span><span><returns></span><span></returns></span></span><span>public</span><span>static</span><span>byte</span>[] <span>GetKeyMd5Hash</span>(<span>string</span> key) { MD5CryptoServiceProvider hashmd5 = <span>new</span> MD5CryptoServiceProvider(); <span>byte</span>[] keyBytes = hashmd5.ComputeHash(UTF8Encoding.UTF8.GetBytes(key)); hashmd5.Clear(); <span>return</span> keyBytes; } <span><span>///</span><span><summary></span></span><span><span>///</span> TripleDES 加密</span><span><span>///</span><span></summary></span></span><span><span>///</span><span><param name="toEncrypt"></span><span></param></span></span><span><span>///</span><span><param name="privateKey"></span><span></param></span></span><span><span>///</span><span><returns></span><span></returns></span></span><span>public</span><span>static</span><span>string</span><span>DesEncrypt</span>(<span>string</span> toEncrypt, <span>byte</span>[] privateKey) { <span>byte</span>[] toEncryptArray = UTF8Encoding.UTF8.GetBytes(toEncrypt); TripleDESCryptoServiceProvider tdes = <span>new</span> TripleDESCryptoServiceProvider { Key = privateKey, Mode = CipherMode.ECB, Padding = PaddingMode.PKCS7 }; ICryptoTransform cTransform = tdes.CreateEncryptor(); <span>byte</span>[] resultArray = cTransform.TransformFinalBlock(toEncryptArray, <span>0</span>, toEncryptArray.Length); tdes.Clear(); <span>return</span> Convert.ToBase64String(resultArray, <span>0</span>, resultArray.Length); } <span><span>///</span><span><summary></span></span><span><span>///</span> TripleDES解密</span><span><span>///</span><span></summary></span></span><span><span>///</span><span><param name="toDecrypt"></span><span></param></span></span><span><span>///</span><span><param name="privateKey"></span><span></param></span></span><span><span>///</span><span><returns></span><span></returns></span></span><span>public</span><span>static</span><span>string</span><span>DesDecrypt</span>(<span>string</span> toDecrypt, <span>byte</span>[] privateKey) { <span>//先base64解密 因为加密的时候最后走了一道base64加密</span><span>byte</span>[] enBytes = Convert.FromBase64String(toDecrypt); TripleDESCryptoServiceProvider tdes = <span>new</span> TripleDESCryptoServiceProvider { Key = privateKey, Mode = CipherMode.ECB, Padding = PaddingMode.PKCS7 }; ICryptoTransform cTransform = tdes.CreateDecryptor(); <span>byte</span>[] resultArray = cTransform.TransformFinalBlock(enBytes, <span>0</span>, enBytes.Length); tdes.Clear(); <span>return</span> Encoding.UTF8.GetString(resultArray); } } } </code>
PHP code
<code><span><span><?php</span><span>/* TripleDES加密 */</span><span><span>function</span><span>DesEncrypt</span><span>(<span>$data</span>)</span> {</span><span>//Pad for PKCS7</span><span>$blockSize</span> = mcrypt_get_block_size(<span>'tripledes'</span>, <span>'ecb'</span>); <span>$len</span> = strlen(<span>$data</span>); <span>$pad</span> = <span>$blockSize</span> - (<span>$len</span> % <span>$blockSize</span>); <span>$data</span> .= str_repeat(chr(<span>$pad</span>), <span>$pad</span>); <span>$key</span> = <span>"home"</span>; <span>$key</span> = md5(<span>$key</span>,<span>TRUE</span>); <span>$key</span> .= substr(<span>$key</span>,<span>0</span>,<span>8</span>); <span>//comment this if you use 168 bits long key</span><span>//Encrypt data</span><span>$encData</span> = mcrypt_encrypt(<span>'tripledes'</span>, <span>$key</span>, <span>$data</span>, <span>'ecb'</span>); <span>return</span> base64_encode(<span>$encData</span>); } <span>/* TripleDES解密 */</span><span><span>function</span><span>DesDecrypt</span><span>(<span>$data</span>)</span> {</span><span>$key</span> = <span>"home"</span>; <span>$key</span> = md5(<span>$key</span>, <span>TRUE</span>); <span>$key</span> .= substr(<span>$key</span>, <span>0</span>, <span>8</span>); <span>//Decrypt data</span><span>$fromBase64Str</span> = base64_decode(<span>$data</span>); <span>$decData</span> = mcrypt_decrypt(<span>'tripledes'</span>, <span>$key</span>, <span>$fromBase64Str</span>, <span>'ecb'</span>); <span>return</span><span>$decData</span>; } <span>/* 测试 */</span><span>$encryptStr</span> = DesEncrypt(<span>"Happy Father's Day!"</span>); <span>echo</span><span>"encrypted string: $encryptStr</br>"</span>; <span>$decryptStr</span> = DesDecrypt(<span>$encryptStr</span>); <span>echo</span><span>"decrypted string: $decryptStr"</span>; <span>?></span></span></code>
このように、PHP は C# で暗号化されたコンテンツを復号することができ、C# も PHP で暗号化されたコンテンツを復号することができます。
どちらも TripleDES 暗号化アルゴリズムですが、C# と PHP の暗号化に使用されるデフォルト モードとパディングは異なります。各言語に付属する TripleDES を直接使用する場合、両者は同じキーを使用して同じ暗号化を実行します。コンテンツが暗号化されている場合、暗号化結果が異なるため、相互に復号化できません。
さまざまな言語で使用される暗号化モードの詳細については、Google を検索してください。 RSA 暗号化アルゴリズムと同様に、C#、JAVA、および PHP は異なるキー形式を使用します。JAVA は ASN を使用し、C# は XML を使用し、PHP は PEM を使用します。その結果、相互に通信できなくなります。
p.s. 3 つの間で RSA の相互運用性を実現する方法は見つかりませんでした。
参考記事:
暗号化アルゴリズムパターンパディング
PHPパディングPKCS7
AES暗号化文字列からPKCS7パディングを追加/削除する方法
PHPと.NETの間でPCSK7とECBを使用したトリプルDES
今日は父の日、最高の幸せな父の日をお祈りします!
').addClass('事前番号付け').hide(); $(this).addClass('has-numbering').parent().append($numbering); for (i = 1; i ').text(i)); }; $numbering.fadeIn(1700); }); });上記では、ph 値の意味を含め、C# と PHP の間の TripleDES 暗号化と復号化が何を意味するのかを紹介しました。これが、PHP チュートリアルに興味のある友人に役立つことを願っています。