Spring AOP 实现 DTO 字段级敏感信息动态脱敏

发布时间 - 2026-01-12 00:00:00    点击率:

本文介绍如何利用 spring aop 在 dto 返回前自动对标注 `@personalinfo` 的字段进行动态脱敏,无需修改业务逻辑或数据库层,通过拦截 getter 方法实现运行时掩码处理。

在构建面向前端的 API 时,常需对敏感字段(如姓名、手机号、身份证号)进行动态脱敏(例如 "张三" → "张*"),且该脱敏行为应与数据持久层解耦——即数据库中仍保存明文,仅在 DTO 序列化为响应体前实时处理。Spring AOP 并不支持直接拦截字段赋值或构造器调用(尤其对 Lombok 生成的无参/全参构造器),但可高效拦截 getter 方法调用:Lombok @Getter 生成的标准 getter(如 getName())属于 public 方法,完全符合 Spring AOP 的代理条件。

✅ 推荐方案:基于 Getter 的环绕通知(Around Advice)

以下为完整实现步骤:

1. 定义脱敏注解(保持不变)

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface PersonalInfo {
    // 可扩展:maskType() default MaskType.STAR;
}

2. 编写 Aspect:拦截所有带 @PersonalInfo 字段的 DTO 的 getter 方法

@Aspect
@Component
public class PersonalInfoAspect {

    @Around("execution(public * *(..)) && " +
            "within(@org.springframework.stereotype.Controller *) && " +
            "target(target) && " +
            "args(..) && " +
            "get(@com.example.annotation.PersonalInfo *)")
    public Object maskPersonalInfo(ProceedingJoinPoint joinPoint) throws Throwable {
        Object result = joinPoint.proceed();
        if (result == null) return result;

        // 获取被调用的 getter 方法名(如 getName → name)
        String methodName = joinPoint.getSignature().getName();
        if (!methodName.startsWith("get") || methodName.length() <= 3) return result;
        String fieldName = Character.toLowerCase(methodName.charAt(3)) + methodName.substring(4);

        // 反射获取目标对象的对应字段是否标注 @PersonalInfo
        Field field = findFieldInClass(joinPoint.getTarget().getClass(), fieldName);
        if (field != null && field.isAnnotationPresent(PersonalInfo.class)) {
            return maskValue(result);
        }
        return result;
    }

    private Field findFieldInClass(Class clazz, String fieldName) {
        try {
            return clazz.getDeclaredField(fieldName);
        } catch (NoSuchFieldException e) {
            if (clazz.getSuperclass() != null) {
                return findFieldInClass(clazz.getSuperclass(), fieldName);
            }
            return null;
        }
    }

    private Object maskValue(Object value) {
        if (value instanceof String str && !str.isBlank()) {
            return str.length() == 1 ? "*" : str.charAt(0) + "*".repeat(str.length() - 1);
        }
        return value; // 其他类型(如 Number)默认不处理,可按需扩展
    }
}
⚠️ 注意事项:Spring AOP 仅代理 Spring 容器管理的 Bean:确保该 Aspect 类被 @Component 扫描,且目标 DTO 被作为返回值由 @Controller 或 @RestController 方法直接返回(Spring MVC 会将其视为代理目标);不适用于非 Spring Bean 场景:若 DTO 是手动 new User() 创建并传入响应体,则无法被 AOP 拦截 —— 此时应改用 @JsonSerialize(Jackson)或 @Schema(OpenAPI)等序列化层脱敏;性能考量:反射查找字段有一定开销,建议配合 ConcurrentHashMap 缓存 Class → Map 提升效率;更健壮的切入点:可将 within(@org.springframework.stereotype.Controller *) 替换为 execution(* com.example.controller..*.*(..)) 显式限定包路径。

3. 使用示例(Controller 层)

@RestController
public class UserController {

    @GetMapping("/user/{id}")
    public User getUser(@PathVariable String id) {
        // 假设从 service 获取原始 User(name 为明文)
        return User.builder()
                .id(id)
                .name("李四丰") // 数据库真实值
                .build();
    }
}

调用 /user/123 将返回:

{ "id": "123", "name": "李**" }

✅ 替代方案对比(供选型参考)

方案 优点 缺点 适用场景
Getter 级 AOP(本文方案) 无侵入、逻辑集中、与序列化解耦 依赖 Spring Bean 生命周期;无法覆盖手动 new 对象 Spring MVC REST API 标准返回流
Jackson @JsonSerialize 精确控制 JSON 输出;支持任意对象;不依赖 Spring 上下文 需为每个字段/类型定义 Serializer;配置分散 微服务间 JSON 通信、DTO 序列化强管控
DTO 构造时脱敏(Builder 模式) 编译期安全、零反射、高性能 业务代码需显式调用,违反单一职责;重复逻辑多 小型项目或合规强要求场景

综上,对于大多数 Spring Boot Web 应用,基于 getter 的 AOP 脱敏是最平衡的选择:它在保持代码简洁性的同时,实现了关注点分离与运行时灵活性。只需确保 DTO 通过 Controller 方法直接返回,即可“零配置”生效。


# js  # 前端  # json  # app  # rest api  # spring mvc  # 回流  # red  # mvc  # spring  # spring boot  # class  # public  # map  # 对象  # 数据库  # 序列化  # 只需  # 有一定  # 将其  # 它在  # 可将  # 高性能  # 数据库中  # 时应  # 应与 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何在云服务器上快速搭建个人网站?  新三国志曹操传主线渭水交兵攻略  详解Oracle修改字段类型方法总结  谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复  Android利用动画实现背景逐渐变暗  Laravel怎么实现软删除SoftDeletes_Laravel模型回收站功能与数据恢复【步骤】  Laravel如何操作JSON类型的数据库字段?(Eloquent示例)  Java垃圾回收器的方法和原理总结  专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?  Laravel怎么在Blade中安全地输出原始HTML内容  如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环  如何快速搭建高效可靠的建站解决方案?  北京企业网站设计制作公司,北京铁路集团官方网站?  C++时间戳转换成日期时间的步骤和示例代码  原生JS实现图片轮播切换效果  如何用AI帮你把自己的生活经历写成一个有趣的故事?  网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?  JavaScript常见的五种数组去重的方式  Laravel如何配置和使用缓存?(Redis代码示例)  头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  php 三元运算符实例详细介绍  Laravel如何实现事件和监听器?(Event & Listener实战)  如何在IIS服务器上快速部署高效网站?  中山网站推广排名,中山信息港登录入口?  Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制  如何彻底删除建站之星生成的Banner?  Laravel怎么进行数据库回滚_Laravel Migration数据库版本控制与回滚操作  胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?  Laravel如何处理JSON字段的查询和更新_Laravel JSON列操作与查询技巧  Laravel Blade模板引擎语法_Laravel Blade布局继承用法  如何在Windows虚拟主机上快速搭建网站?  Laravel如何实现邮件验证激活账户_Laravel内置MustVerifyEmail接口配置【步骤】  动图在线制作网站有哪些,滑动动图图集怎么做?  Python文件异常处理策略_健壮性说明【指导】  php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】  Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录  Laravel如何自定义分页视图?(Pagination示例)  Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑  在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?  Claude怎样写结构化提示词_Claude结构化提示词写法【教程】  Laravel如何编写单元测试和功能测试?(PHPUnit示例)  Bootstrap整体框架之JavaScript插件架构  北京网站制作公司哪家好一点,北京租房网站有哪些?  Swift开发中switch语句值绑定模式  Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践  零服务器AI建站解决方案:快速部署与云端平台低成本实践  Laravel如何创建自定义Artisan命令?(代码示例)  Laravel如何发送系统通知_Laravel Notifications实现多渠道消息通知