AES加密报BadPaddingException主因是密钥长度、填充方式或编码不一致;需确保密钥为16/24/32字节、加解密参数完全相同、IV固定或传输、统一UTF-8编码,并用java.util.Base64安全转换密文。
BadPaddingException
多数情况不是算法写错了,而是密钥长度、填充方式或字符编码不一致导致的。AES要求密钥必须是128/192/256位(即16/24/32字节),用SecretKeySpec构造时若传入的字符串没转成对应字节数组,就容易出错。
MessageDigest对原始字符串SHA-256哈希后取前32字节)Cipher实例化参数,例如"AES/CBC/PKCS5Padding",不能一边用PKCS5Padding,另一边用NoPadding
IV必须固定或随密文一起传输;若用SecureRandom每次生成新IV,解密时没保存它就必然失败StandardCharsets.UTF_8,避免new String(bytes)默认平台编码引发乱码二进制密文不能直接当字符串用,常见做法是Base64编码。但别用sun.misc.BASE64Encoder(已废弃且非标准),改用java.util.Base64。
Base64.getEncoder().encodeToString(cipher.doFinal(plainBytes))
Base64.getDecoder().decode(encodedString)还原为字节数组Base64.getUrlEncoder()
DES加密在Java 21里直接抛NoSuchAlgorithmException
JDK 17+默认禁用DES(含DESede),因为密钥太短、易被暴力破解。OpenJDK和Oracle JDK都已将其列为“insecure”,运行时策略文件java.security里jdk.crypto.disabledAlgorithms会拦截。
KeyGenerator.getInstance("DES"),改用AES
JAVA_HOME/conf/security/java.security,删掉DES相关条目(不推荐)
Cipher实例缓存复用不要缓存Cipher对象。它不是线程安全的,且内部状态(如opmode、iv)会在doFinal()后被清空或改变。
Cipher:用Cipher.getInstance("AES/CBC/PKCS5Padding")
SecretKey和IvParameterSpec(只要它们本身不变)getInstance开销极小,远低于因共享Cipher导致的并发异常或解密失败Cipher.doFinal就完事。