1

C# ベースの復号化関数を Java に変換しました。これは問題なく動作し、C# プログラムによって暗号化されたパスワードを復号化するために使用できます。ソースコードは次のとおりです。

import org.apache.commons.codec.binary.Base64;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import java.security.Key;

public class TestDecrpt {
    public static void main(String[] args) throws Exception {
        String data = "encrypted data";
        String sEncryptionKey = "encryption key";
        byte[] rawData = new Base64().decode(data);
        byte[] salt = new byte[8];
        System.arraycopy(rawData, 0, salt, 0, salt.length);

        Rfc2898DeriveBytes keyGen = new Rfc2898DeriveBytes(sEncryptionKey, salt);

        byte[] IV = keyGen.getBytes(128 / 8);
        byte[] keyByte = keyGen.getBytes(256 / 8);

        Key key = new SecretKeySpec(keyByte, "AES");
        Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
        cipher.init(Cipher.DECRYPT_MODE, key, new IvParameterSpec(IV));
        int pureDataLength = rawData.length - 8;
        byte[] pureData = new byte[pureDataLength];
        System.arraycopy(rawData, 8, pureData, 0, pureDataLength);
        String plaintext = new String(cipher.doFinal(pureData), "UTF-8").replaceAll("\u0000", "");
        System.out.println(plaintext);
    }
}

そのアルゴリズムに従って、暗号化関数を作成します。コードは次のとおりです。

import org.apache.commons.codec.binary.Base64;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import java.security.Key;
import java.security.SecureRandom;


public class testEncrypt {
    public static void main(String[] args) throws Exception {
        String data = "Welcome2012~1@Welcome2012~1@Welcome2012~1@Welcome2012~1@Welcome2012~1@";
        String sEncryptionKey = "encryption key"; # the same key
        byte[] rawData = new Base64().decode(data);
        SecureRandom random = new SecureRandom();
        byte[] salt = new byte[8];
        random.nextBytes(salt);
        Rfc2898DeriveBytes keyGen = new Rfc2898DeriveBytes(sEncryptionKey, salt);

        byte[] IV = keyGen.getBytes(128 / 8);
        byte[] keyByte = keyGen.getBytes(256 / 8);

        Key key = new SecretKeySpec(keyByte, "AES");
        Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
        cipher.init(Cipher.ENCRYPT_MODE, key, new IvParameterSpec(IV));
        byte[] out2 = cipher.doFinal(rawData);

        byte[] out = new byte[8 + out2.length];
        System.arraycopy(salt, 0, out, 0, 8);
        System.arraycopy(out2, 0, out, 8, out2.length);
        //String outStr=new String(out,"UTF-8");
        String outStr = new Base64().encodeToString(out);
        System.out.println(outStr);
        System.out.print(outStr.length());

    }
}

ただし、暗号化されたデータは正しく復号化できず、次のようなガベージ コードが常に返されます。

ꉜ뙧巓妵峩枢要펶땝ꉜ뙧巓妵峩枢枢

暗号化機能に何か問題がありますか?

================================================== ============================== [更新]コードを次のように変更した後

byte[] rawData = data.getBytes("UTF-8");

データは正常に暗号化および復号化できました。ただし、Java で暗号化されたデータは、C# では正しく復号化できませんでした。C# バージョンの復号化関数は次のとおりです。

using System;
using System.IO;
using System.Security.Cryptography;
using System.Text;


namespace Test
{
    class Program
    {
        public static void Main(string[] args)
        {
                string data="EncryptedData";
                string sEncryptionKey="EncryptionKey";

                byte[] rawData = Convert.FromBase64String(data);
                byte[] salt = new byte[8];
                for (int i = 0; i < salt.Length; i++)
                    salt[i] = rawData[i];

                Rfc2898DeriveBytes keyGenerator = new Rfc2898DeriveBytes(sEncryptionKey, salt);
                Rijndael aes = Rijndael.Create();
                aes.IV = keyGenerator.GetBytes(aes.BlockSize / 8);
                aes.Key = keyGenerator.GetBytes(aes.KeySize / 8);

                using (MemoryStream memoryStream = new MemoryStream())
                using (CryptoStream cryptoStream = new CryptoStream(memoryStream, aes.CreateDecryptor(), CryptoStreamMode.Write))
                {
                    cryptoStream.Write(rawData, 8, rawData.Length - 8);
                    cryptoStream.Close();

                    byte[] decrypted = memoryStream.ToArray();
                    Console.Out.WriteLine(Encoding.Unicode.GetString(decrypted));
                    Console.In.ReadLine();
                }

        }
    }
}

元のコードは出力形式として「Unicode」を使用していることがわかりましたが、

Encoding.Unicode.GetString(decrypted)

そのため、Java コードを「Unicode」に変更します。

Java で復号化する場合:

String plaintext = new String(cipher.doFinal(pureData), "Unicode");
System.out.println(plaintext);

Java で暗号化する場合:

byte[] rawData = data.getBytes("Unicode");

しかし、C# コードを使用して、Java プログラムによって暗号化されたデータを復号化しても、まだガベージ コードに遭遇します。

この問題を解決するにはどうすればよいですか? 魔法のトリックはありますか?


[最終更新] 「UTF-8」の代わりに「UTF-16LE」を使用したら、問題はなくなりました。「UTF-16LE」はC#の「Unicode」に相当するJavaのようです。

4

1 に答える 1

3

これが問題です:

String data = "Welcome2012~1@Welcome2012~1@Welcome2012~1@Welcome2012~1@Welcome2012~1@";
byte[] rawData = new Base64().decode(data);

そのテキストは、base64でエンコードされたバイナリデータを意味するものではありません。ただのテキストです。なぜbase64データとしてデコードしようとしているのですか?

あなたが欲しい:

byte[] rawData = data.getBytes("UTF-8");

そうすれば、後で書くときに:

String plaintext = new String(cipher.doFinal(pureData), "UTF-8")
                                    .replaceAll("\u0000", "");

あなたは逆の行動をしています。(確かに、あなたはおそらくreplaceAll電話を必要とすべきではありませんが、それは別の問題です。)

このような場合は、「出て行く」途中の手順が「入っていく」途中の手順の逆であることを確認する必要があります。したがって、正しいコードでは、次のようになります。

Unencrypted text data => unencrypted binary data (encode via UTF-8)
Unencrypted binary data => encrypted binary data (encrypt with AES)
Encrypted binary data => encrypted text data (encode with base64)

したがって、逆に、次のようにします。

Encrypted text data => encrypted binary data (decode with base64)
Encrypted binary data => unencrypted binary data (decrypt with AES)
Unencrypted binary data => unencrypted text data (decode via UTF-8)
于 2012-08-07T03:42:51.637 に答える