2012-06-16 51 views
3

我想加密和解密一些简单的文本。但由于某种原因,我收到了一个奇怪的错误:javax.crypto.BadPaddingException。为什么JCE会生成未正确填充的字节?AES/CBC/PKCS5Padding问题

我有以下代码:

import javax.crypto.Cipher; 
import javax.crypto.KeyGenerator; 
import javax.crypto.SecretKey; 
import javax.crypto.spec.IvParameterSpec; 
import java.security.SecureRandom; 

public class SimplestTest { 
    public static void main(String[] args) throws Exception { 
     SecureRandom rnd = new SecureRandom(); 

     String text = "Hello, my dear! ... " + System.getProperty("user.home"); 
     byte[] textData = text.getBytes(); 

     IvParameterSpec iv = new IvParameterSpec(rnd.generateSeed(16)); 

     KeyGenerator generator = KeyGenerator.getInstance("AES"); 
     generator.init(128); 
     SecretKey k = generator.generateKey(); 

     Cipher c = Cipher.getInstance("AES/CBC/PKCS5Padding"); 
     c.init(Cipher.ENCRYPT_MODE, k, iv); 
     c.update(textData); 
     byte[] data = c.doFinal(); 

     System.out.println("E: " + data.length); 

     c = Cipher.getInstance("AES/CBC/PKCS5Padding"); 
     c.init(Cipher.DECRYPT_MODE, k, iv); 
     c.update(data); 

     System.out.println("R: " + c.doFinal().length); 
    } 

} 

但由于某种原因,这是行不通的。它失败,这个例外:

E: 16 
Exception in thread "main" javax.crypto.BadPaddingException: Given final block not properly padded 
     at com.sun.crypto.provider.SunJCE_f.b(DashoA13*..) 
     at com.sun.crypto.provider.SunJCE_f.b(DashoA13*..) 
     at com.sun.crypto.provider.AESCipher.engineDoFinal(DashoA13*..) 
     at javax.crypto.Cipher.doFinal(DashoA13*..) 
     at SimplestTest.main(SimplestTest.java:31) 

怎么回事?数据大小为16个字节,但仍“未正确填充”?

回答

4

Cipher.update也返回一个字节[]。因此,当您解密时,您缺少部分加密数据。更新最后一节,内容如下:

Cipher c = Cipher.getInstance("AES/CBC/PKCS5Padding"); 
c.init(Cipher.ENCRYPT_MODE, k, iv); 
byte[] someData = c.update(textData); 
byte[] moreData = c.doFinal(); 

System.out.println("E: " + (someData.length + moreData.length)); 

c = Cipher.getInstance("AES/CBC/PKCS5Padding"); 
c.init(Cipher.DECRYPT_MODE, k, iv); 
byte[] someDecrypted = c.update(someData); 
byte[] moreDecrypted = c.doFinal(moreData); 

System.out.println("R: " + (someDecrypted.length + moreDecrypted.length)); 
+0

非常感谢。现在一切工作正常。 –

4

你可以放弃对update呼叫而直接传递byte[]数据doFinal进行加密或一步到位解密的操作。