0

0

AES CBC 加密在 Java 与 Angular 中的跨平台兼容性实践

碧海醫心

碧海醫心

发布时间:2026-02-18 21:10:03

|

526人浏览过

|

来源于php中文网

原创

AES CBC 加密在 Java 与 Angular 中的跨平台兼容性实践

本文详解如何在 angular(cryptojs)与 java 后端间实现 aes/cbc 加密解密的无缝协同,重点解决 pkcs#5 与 pkcs#7 填充不一致、密钥派生参数错配等常见跨语言兼容问题。

本文详解如何在 angular(cryptojs)与 java 后端间实现 aes/cbc 加密解密的无缝协同,重点解决 pkcs#5 与 pkcs#7 填充不一致、密钥派生参数错配等常见跨语言兼容问题。

在前后端分离架构中,使用 AES-CBC 模式进行敏感数据加解密是常见需求。然而,当 Angular 前端采用 CryptoJS、Java 后端使用原生 javax.crypto 时,常因算法细节不一致导致解密失败——最典型的症状即:Java 报 NoSuchPaddingException: PKCS7Padding,而强行改用 PKCS5Padding 后,CryptoJS 又抛出填充验证错误。这并非“PKCS#5 与 PKCS#7 本质不同”,而是两者在 8 字节块场景下行为完全等价,但 Java 标准库仅内置 PKCS5Padding 实现(JDK 不支持 PKCS7Padding 算法名),而 CryptoJS 默认 Pkcs7 命名易引发误解。

✅ 正确做法:统一使用 PKCS#5 填充语义,但保持命名与参数严格对齐

1. Angular(CryptoJS)端关键修正

  • 密钥长度必须为 16 字节(128 bit):keySize: 16(非 128/32=4 或 128/8=16 的错误推导);
  • IV 必须为 16 字节且与 Java 端完全一致:不能直接 parse(clef)(若 clef 长度不足会静默截断或填充);
  • 显式指定 padding: CryptoJS.pad.Pkcs5(虽 CryptoJS 内部 Pkcs5 与 Pkcs7 实现相同,但为明确语义并避免混淆,建议统一用 Pkcs5);
  • 盐值(salt)和 PBKDF2 迭代次数需与 Java 完全一致

修正后的 Angular 加密代码如下:

import { Injectable } from '@angular/core';
import * as CryptoJS from 'crypto-js';

@Injectable({ providedIn: 'root' })
export class CryptoService {
  private readonly SALT = '123456789123'; // 必须与 Java 端完全相同
  private readonly ITERATIONS = 1000;

  encrypt(message: string, password: string): string {
    // 生成固定长度 salt(UTF8 编码)
    const saltBytes = CryptoJS.enc.Utf8.parse(this.SALT);

    // PBKDF2 密钥派生:输出 128-bit (16 byte) 密钥
    const key = CryptoJS.PBKDF2(password, saltBytes, {
      keySize: 16, // ← 关键:16 字节 = 128 bit
      iterations: this.ITERATIONS,
      hasher: CryptoJS.algo.SHA256
    });

    // IV 必须为 16 字节;此处示例使用固定 IV(生产环境应随机生成并传输)
    // 注意:CryptoJS.enc.Utf8.parse() 对短字符串会补零,务必确保输入为 16 字符
    const ivString = 'MnTQLHcWumIKTXpQ'; // 与 Java 端硬编码 IV 严格一致
    const iv = CryptoJS.enc.Utf8.parse(ivString);

    const encrypted = CryptoJS.AES.encrypt(
      CryptoJS.enc.Utf8.parse(message),
      key,
      {
        iv: iv,
        mode: CryptoJS.mode.CBC,
        padding: CryptoJS.pad.Pkcs5 // ← 显式使用 Pkcs5(语义清晰,兼容 Java)
      }
    );

    return encrypted.toString(); // Base64 编码字符串
  }
}

2. Java 端关键修正

  • 算法名必须为 "AES/CBC/PKCS5Padding"(Java 标准库唯一支持的名称);
  • PBEKeySpec 的 keyLength 参数单位是 bit,不是 byte:128(非 128/32=4);
  • 盐值必须用相同字符编码(UTF-8)解析
  • IV 必须与前端完全一致,且类型为 IvParameterSpec

修正后的 Java 解密代码如下:

Lenso.ai
Lenso.ai

AI反向图像搜索

下载
import javax.crypto.*;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.PBEKeySpec;
import java.nio.charset.StandardCharsets;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.KeySpec;
import java.util.Base64;

public class AESUtilService {

    private static final String SALT = "123456789123";
    private static final int ITERATIONS = 1000;
    private static final int KEY_SIZE = 128; // ← 单位:bit(128-bit = 16 bytes)

    public SecretKey getKeyFromPassword(String password)
            throws NoSuchAlgorithmException, InvalidKeySpecException {
        SecretKeyFactory factory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA256");
        // 注意:salt.getBytes() 必须指定 UTF-8,否则编码不一致!
        KeySpec spec = new PBEKeySpec(
                password.toCharArray(),
                SALT.getBytes(StandardCharsets.UTF_8), // ← 显式 UTF-8
                ITERATIONS,
                KEY_SIZE // ← 此处为 128(bit),非 4 或 16
        );
        return factory.generateSecret(spec);
    }

    public String decrypt(String cipherText, SecretKey key)
            throws Exception {
        // ✅ 正确算法名:PKCS5Padding(Java 唯一支持)
        String algorithm = "AES/CBC/PKCS5Padding";

        // IV 必须与前端完全一致(16 字节 ASCII 字符串)
        byte[] ivBytes = "MnTQLHcWumIKTXpQ".getBytes(StandardCharsets.UTF_8);
        IvParameterSpec iv = new IvParameterSpec(ivBytes);

        Cipher cipher = Cipher.getInstance(algorithm);
        cipher.init(Cipher.DECRYPT_MODE, key, iv);

        byte[] decoded = Base64.getDecoder().decode(cipherText);
        byte[] plainBytes = cipher.doFinal(decoded);
        return new String(plainBytes, StandardCharsets.UTF_8);
    }
}

⚠️ 重要注意事项

  • IV 绝不可复用:示例中使用固定 IV 仅用于调试。生产环境必须每次加密生成密码学安全的随机 IV(如 SecureRandom),并将其与密文一同传输(如拼接在 Base64 前缀);
  • 盐值(Salt)可固定:因用于密钥派生,固定 salt 可接受(但需保密);若需更高安全性,可动态生成并随密文传输;
  • 字符编码一致性:前后端所有字符串(salt、password、IV、明文)均需强制使用 UTF-8 编码,避免平台默认编码差异;
  • 密钥长度验证:CryptoJS 的 keySize: 16 表示 16 words(每个 word 32-bit),实际输出密钥字节数 = 16 × 4 = 64 —— 这是常见误区!正确做法是省略 keySize,由 PBKDF2 输出直接作为密钥(CryptoJS 会自动截取前 16 字节),或显式设置 keySize: 4(对应 16 字节)。但更推荐方式是:不设 keySize,让 PBKDF2 输出完整密钥,再手动截取
    const key = CryptoJS.PBKDF2(password, saltBytes, { iterations: 1000, hasher: CryptoJS.algo.SHA256 });
    const aesKey = CryptoJS.enc.Base64.parse(key.toString()).toString(CryptoJS.enc.Latin1).substring(0, 16);

✅ 总结

跨语言 AES-CBC 兼容的核心在于三点统一:填充算法语义(PKCS#5)、密钥派生参数(salt/iterations/keyLength)、IV 字节序列。Java 侧必须使用 "PKCS5Padding",CryptoJS 侧推荐显式使用 Pkcs5 并确保密钥为 16 字节、IV 为 16 字节且 UTF-8 编码一致。忽略任一细节都将导致 BadPaddingException 或解密乱码。遵循本文配置,即可实现稳定、安全的前后端加解密互通。

立即学习Java免费学习笔记(深入)”;

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

553

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

216

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1553

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

640

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

945

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

896

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

185

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

89

2025.08.07

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

561

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.7万人学习

C# 教程
C# 教程

共94课时 | 9.7万人学习

Java 教程
Java 教程

共578课时 | 67.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号