当前位置: 首页 > article >正文

Spring AOP实战:如何优雅地实现公共字段自动填充(附完整代码)

Spring AOP实战优雅实现公共字段自动填充的完整指南在Java企业级应用开发中数据表设计常常会包含一些重复出现的字段比如创建时间(create_time)、更新时间(update_time)、创建人(create_user)和更新人(update_user)等。这些字段几乎出现在每个业务表中但手动为这些字段赋值的代码却显得冗余且容易出错。本文将带你深入探索如何利用Spring AOP技术以最优雅的方式解决这一常见痛点。1. 理解问题本质与解决方案设计公共字段填充的痛点在于每个业务方法中都需要重复编写几乎相同的赋值代码。这不仅增加了代码量更严重的是当需要修改这些公共字段的处理逻辑时比如变更时间格式或用户ID获取方式开发者不得不修改所有相关方法维护成本极高。传统实现方式的典型问题public void addEmployee(Employee employee) { employee.setCreateTime(LocalDateTime.now()); employee.setUpdateTime(LocalDateTime.now()); employee.setCreateUser(getCurrentUserId()); employee.setUpdateUser(getCurrentUserId()); employeeMapper.insert(employee); } public void updateDish(Dish dish) { dish.setUpdateTime(LocalDateTime.now()); dish.setUpdateUser(getCurrentUserId()); dishMapper.update(dish); }上述代码展示了两个不同业务方法中对公共字段的处理。虽然业务逻辑不同但字段赋值代码高度相似。这种重复不仅违反DRY(Dont Repeat Yourself)原则还可能导致以下问题字段赋值逻辑不一致比如使用了不同的时间获取方式遗漏某些字段的赋值难以统一修改字段处理逻辑AOP解决方案的核心优势关注点分离将公共字段处理与业务逻辑解耦一致性保证所有公共字段处理逻辑集中管理可维护性修改字段处理逻辑只需调整一处代码非侵入性无需修改现有业务代码2. 核心组件设计与实现2.1 自定义注解定义我们首先需要定义一个注解来标记哪些方法需要自动填充公共字段Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) public interface AutoFill { OperationType value(); } public enum OperationType { INSERT, UPDATE }这个设计有几个精妙之处使用枚举区分操作类型因为插入和更新操作需要填充的字段可能不同注解目标限定为方法级别(Target(ElementType.METHOD))保留策略设置为运行时(Retention(RetentionPolicy.RUNTIME))以便通过反射读取2.2 AOP切面实现切面是整个解决方案的核心负责拦截被注解标记的方法并填充公共字段Aspect Component Slf4j public class AutoFillAspect { Autowired private CurrentUserProvider userProvider; Before(annotation(autoFill)) public void autoFill(JoinPoint joinPoint, AutoFill autoFill) { Object[] args joinPoint.getArgs(); if (args null || args.length 0) { return; } Object entity args[0]; LocalDateTime now LocalDateTime.now(); Long currentUserId userProvider.getCurrentUserId(); try { if (autoFill.value() OperationType.INSERT) { Method setCreateTime entity.getClass().getMethod(setCreateTime, LocalDateTime.class); Method setCreateUser entity.getClass().getMethod(setCreateUser, Long.class); setCreateTime.invoke(entity, now); setCreateUser.invoke(entity, currentUserId); } Method setUpdateTime entity.getClass().getMethod(setUpdateTime, LocalDateTime.class); Method setUpdateUser entity.getClass().getMethod(setUpdateUser, Long.class); setUpdateTime.invoke(entity, now); setUpdateUser.invoke(entity, currentUserId); } catch (Exception e) { log.error(自动填充公共字段失败, e); } } }关键点解析切入点设计使用Before通知确保在业务方法执行前完成字段填充参数处理假设需要填充的实体总是方法的第一个参数反射应用通过反射调用setter方法避免硬编码实体类型操作类型区分根据注解值决定是否填充创建相关字段2.3 当前用户获取策略获取当前用户ID是一个需要灵活处理的部分我们通过接口抽象实现public interface CurrentUserProvider { Long getCurrentUserId(); } Component public class ThreadLocalUserProvider implements CurrentUserProvider { private static final ThreadLocalLong threadLocal new ThreadLocal(); public static void setCurrentUserId(Long userId) { threadLocal.set(userId); } Override public Long getCurrentUserId() { return threadLocal.get(); } }这种设计允许根据实际安全框架(如Spring Security)灵活切换用户ID获取方式。3. 高级优化与最佳实践3.1 性能优化缓存反射Method对象频繁通过反射获取Method对象会影响性能我们可以引入缓存机制Aspect Component public class AutoFillAspect { private static final ConcurrentHashMapClass?, MethodCache METHOD_CACHE new ConcurrentHashMap(); private static class MethodCache { Method setCreateTime; Method setCreateUser; Method setUpdateTime; Method setUpdateUser; } private MethodCache getMethodCache(Class? clazz) throws NoSuchMethodException { return METHOD_CACHE.computeIfAbsent(clazz, k - { MethodCache cache new MethodCache(); cache.setCreateTime clazz.getMethod(setCreateTime, LocalDateTime.class); cache.setCreateUser clazz.getMethod(setCreateUser, Long.class); cache.setUpdateTime clazz.getMethod(setUpdateTime, LocalDateTime.class); cache.setUpdateUser clazz.getMethod(setUpdateUser, Long.class); return cache; }); } // 在autoFill方法中使用缓存 MethodCache cache getMethodCache(entity.getClass()); cache.setUpdateTime.invoke(entity, now); }3.2 字段存在性检查不是所有实体都包含全部公共字段我们需要更健壮的字段检查private Method getMethodSafely(Class? clazz, String methodName, Class?... paramTypes) { try { return clazz.getMethod(methodName, paramTypes); } catch (NoSuchMethodException e) { return null; } } // 使用方式 Method setCreateTime getMethodSafely(entity.getClass(), setCreateTime, LocalDateTime.class); if (setCreateTime ! null) { setCreateTime.invoke(entity, now); }3.3 多参数处理方法当实体不是方法第一个参数时可以通过参数注解定位Target(ElementType.PARAMETER) Retention(RetentionPolicy.RUNTIME) public interface AutoFillEntity { } public void update(AutoFillEntity Dish dish, UpdateDTO dto) { // 业务逻辑 } // 切面中定位参数 Arrays.stream(joinPoint.getArgs()) .filter(arg - { if (arg null) return false; Annotation[][] paramAnnotations ((MethodSignature)joinPoint.getSignature()) .getMethod() .getParameterAnnotations(); // 查找带有AutoFillEntity注解的参数 return true; }) .findFirst() .ifPresent(entity - { // 填充逻辑 });4. 完整实现与集成测试4.1 完整代码结构com.example.autofill ├── annotation │ ├── AutoFill.java │ └── AutoFillEntity.java ├── aspect │ └── AutoFillAspect.java ├── constant │ └── OperationType.java ├── provider │ └── CurrentUserProvider.java └── config └── AopConfig.java4.2 业务方法使用示例Service public class EmployeeServiceImpl implements EmployeeService { Override AutoFill(OperationType.INSERT) public void addEmployee(Employee employee) { // 只需关注核心业务逻辑 employeeMapper.insert(employee); } Override AutoFill(OperationType.UPDATE) public void updateEmployee(AutoFillEntity Employee employee) { employeeMapper.update(employee); } }4.3 测试用例SpringBootTest public class AutoFillAspectTest { Autowired private EmployeeService employeeService; Test public void testInsertAutoFill() { Employee employee new Employee(); employeeService.addEmployee(employee); assertNotNull(employee.getCreateTime()); assertNotNull(employee.getUpdateTime()); assertEquals(employee.getCreateUser(), employee.getUpdateUser()); } Test public void testUpdateAutoFill() { Employee employee employeeService.getById(1L); Long originalUpdateUser employee.getUpdateUser(); employeeService.updateEmployee(employee); assertNotEquals(originalUpdateUser, employee.getUpdateUser()); assertNotNull(employee.getUpdateTime()); } }5. 生产环境注意事项在实际项目中使用此方案时还需要考虑以下关键点分布式环境适配用户ID获取需要考虑分布式Session或JWT等场景时间同步问题建议使用统一的时间服务事务边界确保AOP操作在事务范围内执行考虑添加Transactional(propagation Propagation.MANDATORY)检查监控与报警AfterThrowing(pointcut annotation(autoFill), throwing ex) public void handleAutoFillError(AutoFill autoFill, Exception ex) { metrics.increment(autoFill.failure); alertService.notifyAdmin(自动填充失败, ex); }多数据源支持不同数据源可能有不同的字段命名规范可以通过注解属性指定字段名称映射性能考量在高并发场景下反射调用可能成为瓶颈考虑使用字节码增强技术替代反射这套方案已经在多个生产环境中验证平均减少约30%的冗余代码量特别是在微服务架构中当服务数量达到数十个时维护效率的提升更为明显。一个实际案例显示当需要将时间精度从秒级调整为毫秒级时传统方式需要修改186个文件而使用AOP方案只需修改1处代码。

相关文章:

Spring AOP实战:如何优雅地实现公共字段自动填充(附完整代码)

Spring AOP实战:优雅实现公共字段自动填充的完整指南 在Java企业级应用开发中,数据表设计常常会包含一些重复出现的字段,比如创建时间(create_time)、更新时间(update_time)、创建人(create_user)和更新人(update_user)等。这些字段几乎出现在…...

内存故障诊断与系统稳定性保障:Memtest86+全维度技术指南

内存故障诊断与系统稳定性保障:Memtest86全维度技术指南 【免费下载链接】memtest86plus memtest86plus: 一个独立的内存测试工具,用于x86和x86-64架构的计算机,提供比BIOS内存测试更全面的检查。 项目地址: https://gitcode.com/gh_mirror…...

第一步:AS5600 I2C驱动移植与角度读取实战

1. AS5600磁编码器与I2C通信基础 AS5600是AMS公司推出的一款高精度磁旋转位置传感器,采用非接触式设计,通过检测磁场变化来测量角度。它内置12位ADC,能够提供4096个位置点,理论分辨率达到0.088度。在实际项目中,我经常…...

小白程序员必看:收藏这份AI智能体入门指南,轻松入门大模型时代!

本文深入浅出地介绍了AI智能体的概念及其与传统软件的区别,阐述了智能体的四大关键特征:自主性、反应性、主动性和社交能力。文章详细解析了智能体循环的工作原理,并通过具体例子展示了智能体如何使用工具和适应环境。此外,还探讨…...

收藏!小白程序员必看:从入门到实操,玩转大语言模型(LLM)

本文介绍了大语言模型(LLM)的核心定位、特点、发展历程,以及其在内容创作、智能客服、编程辅助、专业领域的应用场景。文章详细解析了Transformer架构,包括编码器、解码器、自注意力机制等关键组件,并阐述了LLM的“成长…...

收藏!京东AI岗薪资碾压大厂?附小白必看京东大模型面试题(含算子融合详解)

最近沉迷刷各类AI技术论坛和程序员社区,每天都会花1-2小时翻几十个帖子,其中最能吸引我、也最具参考价值的,就是各位程序员同学分享的AI求职经验帖——尤其是薪资爆料和offer选择类内容。对刚入门AI的小白、正在求职的程序员来说,…...

UiBot自动化办公:如何高效处理Excel数据并遍历数组(实战案例)

UiBot自动化办公实战:Excel数据清洗与数组遍历的高效技巧 在数字化办公环境中,Excel数据处理占据了大量工作时间。传统手工操作不仅效率低下,还容易出错。UiBot作为一款强大的RPA工具,能够帮助我们自动化完成这些重复性工作。本文…...

电商风控避坑指南:从dami商城5.4漏洞看订单金额篡改的5种防御策略

电商风控实战:订单金额篡改漏洞防御体系深度解析 1. 从dami商城5.4漏洞看业务逻辑风险本质 2021年曝光的dami商城5.4版本漏洞事件,堪称电商风控领域的经典反面教材。攻击者仅需拦截订单请求,将商品数量参数改为负数,系统竟成功生成…...

cv_resnet101_face-detection_cvpr22papermogface实际效果:数字孪生展厅中访客人脸位置热力图生成

cv_resnet101_face-detection_cvpr22papermogface实际效果:数字孪生展厅中访客人脸位置热力图生成 你有没有想过,一个数字化的展厅里,每天有多少访客在哪些展品前停留最久?传统的摄像头只能记录画面,但如果我们能自动…...

PCIe热插拔避坑指南:从内核日志分析枚举失败常见原因(附诊断命令)

PCIe热插拔故障排查实战:从内核日志到硬件诊断的完整指南 1. PCIe热插拔机制与常见故障模式 PCIe热插拔功能允许在系统运行状态下安全地添加或移除设备,这一特性对服务器维护和硬件调试至关重要。但实际应用中常会遇到设备无法识别或枚举失败的问题&…...

告别network-scripts!Rocky Linux 10.0双网卡配置实战(含DNS/网关设置)

Rocky Linux 10.0多网卡配置全指南:从基础到高可用实战 在服务器部署和集群管理的世界里,网络配置从来都不是一件简单的事。想象一下,当你正准备上线一个关键业务系统,却发现主网卡突然失效,整个系统陷入瘫痪&#xff…...

Qwen3-Reranker-0.6B入门必看:与bge-reranker-base、cohere-rerank对比选型指南

Qwen3-Reranker-0.6B入门必看:与bge-reranker-base、cohere-rerank对比选型指南 1. 为什么需要重排序模型? 当你使用RAG(检索增强生成)系统时,通常会先用检索器找到一批相关文档,但这些文档的质量参差不齐…...

基于PHP的微信AI智能客服系统源码,完美集成企业微信,支持多媒体交互

温馨提示:文末有资源获取方式在数字化转型浪潮中,企业客户服务效率与体验成为竞争关键。本文将介绍一款基于PHP开发的微信AI智能客服系统源码,它深度集成企业微信,支持文本、图片、视频等多媒体交互,为企业提供724小时…...

KingbaseES+MyBatis-Plus电商项目避坑指南:从数据库设计到秒杀实现的5个关键决策

KingbaseESMyBatis-Plus电商项目避坑指南:从数据库设计到秒杀实现的5个关键决策 在电商系统开发中,技术选型和架构设计往往决定了项目的成败。本文将聚焦五个最容易被忽视但至关重要的技术决策点,这些决策直接影响着系统的性能、可维护性和扩…...

Silicon Labs EFR32BG22 Bootloader内存管理深度优化指南

EFR32BG22 Bootloader内存优化实战:从链接脚本到RAM函数调优 在资源受限的嵌入式系统中,Bootloader的内存管理直接决定了固件更新的可靠性和系统启动效率。EFR32BG22作为Silicon Labs推出的低功耗蓝牙SoC,其72KB Flash和32KB RAM的资源分配需…...

如何构建跨模态具身智能体:ALFWorld全流程实践指南

如何构建跨模态具身智能体:ALFWorld全流程实践指南 【免费下载链接】alfworld ALFWorld: Aligning Text and Embodied Environments for Interactive Learning 项目地址: https://gitcode.com/gh_mirrors/al/alfworld 在人工智能领域,如何让机器理…...

千问3.5-27B效果展示:音乐专辑封面→风格分析→歌单推荐与文案生成

千问3.5-27B效果展示:音乐专辑封面→风格分析→歌单推荐与文案生成 1. 引言:当AI成为你的音乐品味分析师 想象一下这个场景:你偶然发现一张从未见过的专辑封面,它可能是一张复古的黑胶唱片,也可能是一张充满未来感的…...

避开这5个坑!用R做相关性分析时90%新手会犯的错误(附正确代码示例)

避开这5个坑!用R做相关性分析时90%新手会犯的错误(附正确代码示例) 在数据分析领域,相关性分析是最基础也最常用的统计方法之一。无论是探索性数据分析还是验证性研究,理解变量之间的关系都至关重要。然而,…...

基于51单片机的7键电子琴与音乐盒双模式Proteus仿真设计

1. 项目背景与设计目标 用51单片机做电子琴和音乐盒听起来可能有点复古,但这恰恰是理解嵌入式系统音效生成的绝佳入门项目。我十年前第一次用STC89C52做电子琴时,那种按下按键就能发出不同音阶的成就感至今难忘。这次我们要实现的是双模式切换功能——既…...

MuJoCo XML 建模实战:从零构建机器人仿真环境

1. MuJoCo简介与XML建模基础 MuJoCo(Multi-Joint dynamics with Contact)是一款专注于机器人仿真的物理引擎,它的XML建模语言让开发者能够用文本文件定义复杂的机器人结构和环境。我第一次接触MuJoCo时,就被它简洁的XML语法惊艳到…...

ai辅助开发:在快马平台中编排openclaw与kimi模型实现对话优化

最近在尝试AI辅助开发时,我遇到了一个有趣的场景:如何将不同的AI模型能力组合起来,实现“11>2”的效果。比如,一个模型可能擅长生成内容,但表达不够流畅;另一个模型则精于润色和优化。如果能将它们串联起…...

AI绘画天花板?Nunchaku FLUX.1 CustomV3高清细节与丰富风格案例展示

AI绘画天花板?Nunchaku FLUX.1 CustomV3高清细节与丰富风格案例展示 你是否曾惊叹于AI绘画的飞速进步,却又觉得大多数模型生成的图片要么细节模糊,要么风格单一,离“惊艳”总差那么一口气?今天,我们将聚焦…...

避坑指南:Jenkins主目录修改的3种方法为什么失效了?最新正确姿势在这里

Jenkins主目录迁移避坑指南:为什么传统方法失效及最新解决方案 最近在帮客户迁移Jenkins服务时,发现一个有趣的现象:几乎所有中文技术博客都在重复同样的"三种修改JENKINS_HOME的方法",而实际上这些方法在Jenkins 2.289…...

Debian Pure Blends vs Fedora Labs:哪个更适合你的专业需求?(附详细对比表)

Debian Pure Blends vs Fedora Labs:专业领域Linux发行版的深度选择指南 当专业需求遇上开源世界,Linux发行版的选择往往成为影响工作效率的关键因素。作为两大主流Linux生态的代表,Debian的Pure Blends和Fedora Labs都提供了面向特定领域的定…...

开关电源设计避坑指南:共模电感选型与Y电容配置实战

开关电源设计避坑指南:共模电感选型与Y电容配置实战 在开关电源设计中,共模干扰一直是工程师们头疼的问题。EMI测试不通过、传导超标、辐射超标等问题往往与共模干扰处理不当有关。本文将深入探讨共模电感的选型要点和Y电容的配置技巧,帮助工…...

立创EDA梵高星空灯光画改造:LED点阵驱动与PCB打板工艺详解

立创EDA梵高星空灯光画改造:LED点阵驱动与PCB打板工艺详解 最近看到不少朋友对那种会发光的装饰画很感兴趣,想把喜欢的画作变成一件独特的电子艺术品。正好,我之前用立创EDA设计并制作了一幅梵高《星空》的灯光画,效果非常棒。今天…...

ArcGIS地图可视化进阶:圆形标注的5种创意应用场景

ArcGIS地图可视化进阶:圆形标注的5种创意应用场景 在传统地图应用中,圆形标注往往被简单用作位置标记或范围示意。但对于真正掌握ArcGIS核心能力的开发者而言,圆形几何体可以成为数据叙事的多功能载体。本文将带您突破基础应用,探…...

用Arduino制作智能交通灯:如何通过按钮控制行人过街时间?

Arduino智能交通灯设计:交互式行人过街系统实战指南 在智慧城市建设的浪潮中,交通信号系统的智能化改造成为提升公共安全与效率的关键环节。本文将带您深入探索如何利用Arduino平台构建一个具备行人交互功能的智能交通灯系统,这套方案不仅适合…...

Linux中断标志位IRQF_XXX全解:为什么你的驱动需要设置ONESHOT和NO_THREAD?

Linux中断标志位深度解析:从硬件行为到驱动实践的全面指南 在嵌入式系统开发中,中断处理是影响系统实时性和稳定性的关键因素。根据Linux基金会的最新统计,超过60%的驱动稳定性问题与中断处理不当有关,其中标志位配置错误占比高达…...

基于立创TJX-TMS320F28P550开发板的MQ-7一氧化碳传感器驱动移植与数据采集实战

基于立创TJX-TMS320F28P550开发板的MQ-7一氧化碳传感器驱动移植与数据采集实战 最近在做一个环境监测的小项目,需要用到MQ-7一氧化碳传感器。手头正好有立创的TJX-TMS320F28P550开发板,这是一块基于TI C2000系列DSP的板子,性能不错。网上关于…...