使用 cryptojs 进行 aes 加密时,若未正确处理编码格式,会导致加密结果每次不同、解密后显示十六进制字符串(如 `6869`)而非原始文本,本文详解原因与规范用法。
在 JavaScript 中使用 CryptoJS 实现 AES 加密/解密时,初学者常遇到两个典型问题:
根本原因在于:CryptoJS 的 AES.encrypt() 默认使用随机生成的 128 位 salt 和 IV(初始向量),并采用 OpenSSL 兼容格式封装——这保证了安全性(防止重放攻击),但也意味着即使密钥和明文完全相同,每次加密输出的 Base64 字符串也不同。这是正常且预期的行为,并非 bug。
而第二个问题更常见:CryptoJS.AES.decrypt() 返回的是 CipherParams 对象,其 .toString() 方法默认以十六进制(hex)格式输出原始字节。例如 "hi" 的 UTF-8 编码为 0x68 0x69,因此直接 .toString() 就得到 "6869"。
✅ 正确做法是:解密后显式指定字符编码,通常为 UTF-8:
const encrypted = CryptoJS.AES.encrypt("hi", "12345").toString(); // 得到 Base64 密文
const decrypted = CryptoJS.AES.decrypt(encrypted, "12345");
const plaintext = decrypted.toString(CryptoJS.enc.Utf8); // ✅ 关键:指定 Utf8 编码
console.log(plaintext); // 输出: "hi"⚠️ 注意事项:
const iv = CryptoJS.enc.Utf8.parse("1234567890123456"); // 固定 IV(仅测试用!)
const encrypted = CryptoJS.AES.encrypt("hi", "12345", { iv });? 总结:
免裸字符串密钥,优先使用派生密钥或 WordArray;