【发布时间】:2017-08-25 05:40:12
【问题描述】:
我正在尝试兼容 C# 和 Java 的加密/解密。
据我所知,Java 中的默认模式是“ecb/pkcs5”,而 C# 中的默认模式是“cbc/pkcs7”。
所以我匹配这些东西。
第一个问题是PKCS7和PKCS5兼容吗??,
Java 中没有 PKCS7,所以我使用 PKCS5。但我可以获得相同的加密数据 [即使填充方式不同,pkcs7/pkcs5,] 可能吗?还是这些兼容?
第二个问题是,为什么我得到相同的结果,即使模式,方式都不同?
我比较了“DES-ECB / DES-CBC / TripleDES-ECB”这些东西。和 C# 运行良好,结果都不同。
输入 > HELLO 输出 > (ECB)/dZf3gUY150= (CBC) V17s5QLzynM= (三)sWGS0GMe1jE
但我在 Java 中得到了同样的结果..
输入 > HELLO 输出 > (ECB)/dZf3gUY150= (CBC)/dZf3gUY150= (三)/dZf3gUY150=
调试的时候流程是对的。
这是我的代码。
C#
public static string Encrypt_DES(string originalString, byte[] key, string mode)
{
DESCryptoServiceProvider cryptoProvider = new DESCryptoServiceProvider();
if (mode.Equals("ECB"))
cryptoProvider.Mode = CipherMode.ECB;
else if (mode.Equals("CBC"))
{
cryptoProvider.Mode = CipherMode.CBC;
cryptoProvider.IV = new byte[] { 0, 0, 0, 0, 0, 0, 0, 0 };
}
cryptoProvider.Padding = PaddingMode.PKCS7;
MemoryStream memoryStream = new MemoryStream();
CryptoStream cryptoStream = new CryptoStream(memoryStream, cryptoProvider.CreateEncryptor(key, key), CryptoStreamMode.Write);
StreamWriter writer = new StreamWriter(cryptoStream);
writer.Write(originalString);
writer.Flush();
cryptoStream.FlushFinalBlock();
writer.Flush();
return Convert.ToBase64String(memoryStream.GetBuffer(), 0, (int)memoryStream.Length);
}
public static string Encrypt_TripleDES(string source, string key)
{
TripleDESCryptoServiceProvider desCryptoProvider = new TripleDESCryptoServiceProvider();
MD5CryptoServiceProvider hashMD5Provider = new MD5CryptoServiceProvider();
byte[] byteHash;
byte[] byteBuff;
byteHash = hashMD5Provider.ComputeHash(Encoding.UTF8.GetBytes(key));
desCryptoProvider.Key = byteHash;
desCryptoProvider.Mode = CipherMode.ECB; //CBC, CFB
desCryptoProvider.Padding = PaddingMode.PKCS7;
byteBuff = Encoding.UTF8.GetBytes(source);
string encoded = Convert.ToBase64String(desCryptoProvider.CreateEncryptor().TransformFinalBlock(byteBuff, 0, byteBuff.Length));
return encoded;
}
Java(Android)
public String Encrypt(String str, String desKey, String mode) {
try {
KeySpec keySpec = null;
SecretKey key = null;
Cipher ecipher = null;
if (desKey.length() == 8) {
keySpec = new DESKeySpec(desKey.getBytes("UTF8"));
key = SecretKeyFactory.getInstance("DES").generateSecret(keySpec);
if(mode.equals(ECB)){
ecipher = Cipher.getInstance("DES/ECB/PKCS5Padding");
ecipher.init(Cipher.ENCRYPT_MODE, key);
}else if (mode.equals(CBC)){
ecipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
AlgorithmParameterSpec ivSpec = new IvParameterSpec(ivBytes);
ecipher.init(Cipher.ENCRYPT_MODE, key,ivSpec);
}
} else if (desKey.length() == 24) {
keySpec = new DESedeKeySpec(desKey.getBytes("UTF8"));
key = SecretKeyFactory.getInstance("DESede").generateSecret(keySpec);
ecipher = Cipher.getInstance("DESede/ECB/PKCS5Padding");
ecipher.init(Cipher.ENCRYPT_MODE, key);
}
byte[] data = str.getBytes("UTF-8");
byte[] crypt = ecipher.doFinal(data);
return Base64.encodeToString(crypt, 0);
} catch (Exception ex) {
Log.d("ZVM", ex.getMessage());
}
return null;
}
- 据我了解,'IV' 用于 CBC,在制作密码时,它与 IV 混合(不是密钥,而是类似密钥)。这样对吗?
谢谢。
【问题讨论】:
-
您的第一个问题已回答here(它们是相同的)。第二个问题不清楚,因为您没有显示代码。也许您忘记指定 IV,它是一个随机化器。解密时需要使用相同的 IV,但它不是秘密的。我不知道你在第三个问题中要问什么。如果您在 3DES 中使用的三个子项不同,则 DES 和三重 DES 是不同的。
-
请edit您的问题更清楚。示例输入和输出将有很大帮助。
-
@ArtjomB。先谢谢你的回答。我编辑了很多内容。据我了解您的链接,PKCS5 和 7 都不兼容...对吗?
-
@ArtjomB。我为 CBC 添加了“IV”,然后面临其他错误。我认为“IV”是使用 CBC 模式的必要值。但我不知道我在其他方面错过了什么(欧洲央行除外)......谢谢
标签: java c# android encryption des