javax.crypto.IllegalBlockSizeException : Input length not multiple of 8 bytes

标签 java des tripledes

我在解密时遇到了这个错误,我浏览了类似的帖子,但我没有从那里得到任何帮助。我想将一个对象直接存储在一个加密的文件中,因为我已经发布了我的问题 here .但是在使用流时,我得到了与使用字符串相同的错误。

package security;

import java.security.spec.KeySpec;

import javax.crypto.Cipher;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.DESedeKeySpec;
import sun.misc.BASE64Decoder;
import sun.misc.BASE64Encoder;

/**
 * This class defines methods for encrypting and decrypting using the Triple DES
 * algorithm and for generating, reading and writing Triple DES keys. It also
 * defines a main() method that allows these methods to be used from the command
 * line.
 */
public class TripleDesEncryptionDecryption {
  /**
   * The program. The first argument must be -e, -d, or -g to encrypt,
   * decrypt, or generate a key. The second argument is the name of a file
   * from which the key is read or to which it is written for -g. The -e and
   * -d arguments cause the program to read from standard input and encrypt or
   * decrypt to standard output.
   */
    private static final String UNICODE_FORMAT = "UTF-8";
    public static final String DESEDE_ENCRYPTION_SCHEME = "DESede";
    private KeySpec myKeySpec;
    private SecretKeyFactory mySecretKeyFactory;
    private Cipher cipher;
    byte[] keyAsBytes;
    private String myEncryptionKey;
    private String myEncryptionScheme;
    SecretKey key;
    static String stringToEncrypt="";

    public void setKey(String myKey) throws Exception
    {
        myEncryptionKey = myKey ;
        myEncryptionScheme = DESEDE_ENCRYPTION_SCHEME;
        keyAsBytes = myEncryptionKey.getBytes(UNICODE_FORMAT);
        myKeySpec = new DESedeKeySpec(keyAsBytes);
        mySecretKeyFactory = SecretKeyFactory.getInstance(myEncryptionScheme);
        cipher = Cipher.getInstance("DESede/ECB/NoPadding");
        key = mySecretKeyFactory.generateSecret(myKeySpec);
    }

    /**
     * Method To Encrypt The String
     */
    public String encrypt(byte[] plainText) {
        String encryptedString = null;
        try {
            cipher.init(Cipher.ENCRYPT_MODE, key);
            //byte[] plainText = unencryptedString.getBytes(UNICODE_FORMAT);
            //byte[] encryptedText = cipher.doFinal(plainText);
            BASE64Encoder base64encoder = new BASE64Encoder();
            encryptedString = base64encoder.encode(plainText);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return encryptedString;
    }
    /**
     * Method To Decrypt An Ecrypted String
     */
    public String decrypt(String encryptedString) {
        String decryptedText=null;
        try {
            cipher.init(Cipher.DECRYPT_MODE, key);
            BASE64Decoder base64decoder = new BASE64Decoder();
            System.out.println(myEncryptionKey);
            byte[] encryptedText = base64decoder.decodeBuffer(encryptedString);
            byte[] plainText = cipher.doFinal(encryptedText);
            decryptedText= bytes2String(plainText);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return decryptedText;
    }
    /**
     * Returns String From An Array Of Bytes
     */
    private static String bytes2String(byte[] bytes) {
        StringBuffer stringBuffer = new StringBuffer();
        for (int i = 0; i < bytes.length; i++) {
            stringBuffer.append((char) bytes[i]);
        }
        return stringBuffer.toString();
    }

    /**
     * Testing The DESede Encryption And Decryption Technique
    */
    public static void main(String args []) throws Exception
    {
        TripleDesEncryptionDecryption myEncryptor= new TripleDesEncryptionDecryption();


        myEncryptor.setKey("tarunvermacdac@gmail.com") ;

        System.out.println("tarun1234".getBytes());
        String encrypted=myEncryptor.encrypt("tarun".getBytes());
        String decrypted=myEncryptor.decrypt(encrypted);

        System.out.println("String To Encrypt: "+stringToEncrypt);
        System.out.println("Encrypted Value :" + encrypted);
        System.out.println("Decrypted Value :"+decrypted);
    }
}

最佳答案

我不完全知道你想要实现什么,但我至少可以解释你遇到的错误。

DES3DES block 长度64 位。这意味着您必须将长度是 64 位的倍数纯文本 传递给加密函数。为此,您通常使用特定模式的数据填充/填充明文的最后一个 block ,以便您可以在解密后轻松将其再次删除。

在您的代码中,您可以像这样指定加密参数:DESede/ECB/NoPadding
因此,您明确选择不自动应用任何填充。


要解决此问题,只需指定填充模式(例如 PKCS5Padding)而不是 NoPadding

注意:密码模式 ECB 根本不安全!你应该更好地使用:

Cipher.getInstance("DESede/CBC/PKCS5Padding"); // or "AES" instead of "DESede"

(您必须提供额外的 IV,但在使用 CBCCTR 等模式时)

关于javax.crypto.IllegalBlockSizeException : Input length not multiple of 8 bytes,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25425486/

相关文章:

c++ - DES。弱键!

python - Pyocrypt DES3文件加密,解密缺失部分文本

php - 在 PHP 中使用 PBEWithMD5AndTripleDES 解密

c# - C# 和 Java 中 Triple DES 的实现有什么区别吗? Java 给出错误 IV 大小错误

java - 为什么 w3c.dom.Element 是按照字母顺序设置属性的?

java - 为 glsl 准备模型、 View 和投影矩阵

java - 小于 ObjectOutputStream 基元的基元数组

java - 寻找 DES key

php - 在 VB.NET 中加密并在 PHP 中解密

java - Spring Security 在提供自定义登录表单时是否需要自定义登录 Controller ?