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 resu
lt;
// 获取被调用的 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实现多渠道消息通知


lt;
// 获取被调用的 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)默认不处理,可按需扩展
}
}