我尝试通过这种方式加密我的文件:
加密
static void encrypt(String strInput , String strOutput) throws IOException,
NoSuchAlgorithmException,NoSuchPaddingException, InvalidKeyException {
FileInputStream fis = new FileInputStream(strInput);
FileOutputStream fos = new FileOutputStream(strOutput);
SecretKeySpec sks = new SecretKeySpec("MyDifficultPassw".getBytes(),
"AES");
Cipher cipher = Cipher.getInstance("AES");
cipher.init(Cipher.ENCRYPT_MODE, sks);
CipherOutputStream cos = new CipherOutputStream(fos, cipher);
int b;
byte[] d = new byte[8];
while ((b = fis.read(d)) != -1) {
cos.write(d, 0, b);
}
// Flush and close streams.
cos.flush();
cos.close();
fis.close();
}
并通过以下方式解密:
解密:
static String decrypt(String strInput) throws IOException, NoSuchAlgorithmException,
NoSuchPaddingException, InvalidKeyException {
FileInputStream fis = new FileInputStream(strInput);
int endFile = strInput.length() - 4;
String strOut = strInput.substring(0, endFile) + "xx.jpg";
FileOutputStream fos = new FileOutputStream(strOut);
SecretKeySpec sks = new SecretKeySpec("MyDifficultPassw".getBytes(),
"AES");
Cipher cipher = Cipher.getInstance("AES");
cipher.init(Cipher.DECRYPT_MODE, sks);
CipherInputStream cis = new CipherInputStream(fis, cipher);
int b;
byte[] d = new byte[8];
while ((b = cis.read(d)) != -1) {
fos.write(d, 0, b);
}
fos.flush();
fos.close();
cis.close();
return strOut;
}
但是,结果文件的大小为0 kb,当我尝试在解密时对b = cis.read(d)
进行故障排除时,总是返回-1,同时cis.available()
总是返回0.有人可以告诉我我的代码的哪一部分是错的?
注意:我可以确保要解密的文件始终存在。
答案 0 :(得分:1)
我认为此问题是因为您尝试解密未加密(或未正确加密)的数据。
在decrypt()
方法中,CipherOutputStream
隐藏了Cipher
类可能抛出的所有异常。见javadoc for CipherOutputStream
:
此外,此类捕获其祖先类未抛出的所有异常。
要公开此问题,您可能需要手动实施密码使用。这是一个简单的例子:
static String decrypt(String strInput) throws IOException,
NoSuchAlgorithmException, NoSuchPaddingException,
InvalidKeyException, IllegalBlockSizeException, BadPaddingException {
FileInputStream fis = new FileInputStream(strInput);
int endFile = strInput.length() - 4;
String strOut = strInput.substring(0, endFile) + "xx.txt";
FileOutputStream fos = new FileOutputStream(strOut);
SecretKeySpec sks = new SecretKeySpec("MyDifficultPassw".getBytes(), "AES");
Cipher cipher = Cipher.getInstance("AES");
cipher.init(Cipher.DECRYPT_MODE, sks);
int b;
byte[] d = new byte[8];
while ((b = fis.read(d)) != -1) {
fos.write(cipher.update(d));
}
fos.write(cipher.doFinal());
fos.flush();
fos.close();
fis.close();
return strOut;
}
您在问题中发布的算法似乎适用于有效输入。例如,让我们假设以下主要内容:
public static void main(String[] argv) {
try {
encrypt("test.txt", "XXX.txt");
decrypt("XXX.txt");
}
catch (Exception e) {
System.out.println(e);
e.printStackTrace();
}
}
使用它,并使用文本文件和JPG文件进行测试,您的算法执行完美。但是,当使用无效输入到解密算法时,您描述的问题就开始出现了。
对于测试,让我们想象一下,我们在尝试解密明确的文件时出现“错误”(只需更改decrypt()
中传递给main
的参数:
encrypt("test.txt", "XXX.txt");
decrypt("test.txt");
然后当然对decrypt()
方法的输入填充是错误的,我们应该得到一个例外。
然而,使用您的decrypt()
版本,没有例外。我们得到的只是一个空文件。
使用上面显示的decrypt()
方法的修改版本,我们得到以下异常:
javax.crypto.BadPaddingException: Given final block not properly padded
javax.crypto.BadPaddingException: Given final block not properly padded
at com.sun.crypto.provider.CipherCore.doFinal(CipherCore.java:811)
at com.sun.crypto.provider.CipherCore.doFinal(CipherCore.java:676)
at com.sun.crypto.provider.AESCipher.engineDoFinal(AESCipher.java:313)
at javax.crypto.Cipher.doFinal(Cipher.java:1970)
at MainTest.decrypt(MainTest.java:71)
at MainTest.main(MainTest.java:21)