使用AOP优化Spring Boot Controller参数:自动填充常用字段的技巧

欢迎来到我的博客,代码的世界里,每一行都是一个故事


在这里插入图片描述

🎏:你只管努力,剩下的交给时间

🏠 :小破站

使用AOP优化Spring Boot Controller参数:自动填充常用字段的技巧

    • 前言
    • 为什么使用AOP
      • 为什么使用AOP实现参数重写?
        • 好处:
        • 坏处:
      • 其他实现方式及其比较
        • 1. Controller基类方法
        • 2. 使用拦截器(Interceptor)
      • 总结
    • 实现
      • 解释说明:

前言

在现代Web开发中,通过AOP实现参数重写是一种高效且优雅的方式。它不仅能帮助开发者简化重复性的代码编写,还能有效提升接口的安全性和可靠性。今天,我们将探索如何利用AOP技术,在Spring Boot项目中实现对Controller保存方法参数的智能填充,让你的API开发更加高效和愉快。

为什么使用AOP

理解为什么选择使用AOP(面向切面编程)来实现参数重写是很重要的,同时还可以考虑其他实现方式。下面我们来详细探讨一下这些方面:

为什么使用AOP实现参数重写?

好处:
  1. 解耦和增强可维护性:AOP可以将横切逻辑(如参数填充)从业务逻辑中分离出来,避免代码重复,提高代码的清晰度和可维护性。
  2. 集中管理和复用:通过AOP,可以集中管理和配置参数填充逻辑,使得多个Controller方法都能够共享同一段逻辑,减少重复开发。
  3. 方便扩展和修改:当需要修改或扩展参数填充逻辑时,只需调整AOP切面,而不必修改每个涉及到参数填充的Controller方法。
坏处:
  1. 引入复杂性:AOP的使用可能会增加代码的复杂性,特别是对于初学者来说,理解和调试AOP可能会有一定的学习曲线。
  2. 运行时性能开销:AOP通常在运行时动态生成代理对象或者织入代码,可能会对系统性能产生一定的影响,尤其是在大规模应用中。

其他实现方式及其比较

1. Controller基类方法

通过创建一个基类Controller,其中包含公共的参数填充逻辑,所有Controller继承这个基类,实现参数填充的共享。

  • 好处:简单直接,无需引入AOP框架,易于理解和维护。
  • 坏处:如果项目中有多种不同的参数填充逻辑,可能会导致基类代码过于复杂和臃肿。
2. 使用拦截器(Interceptor)

在Spring MVC中,可以通过实现HandlerInterceptor接口,重写preHandle方法,在请求进入Controller方法之前进行参数的预处理。

  • 好处:与AOP类似,可以实现对请求的拦截和处理,但更加灵活,可以针对特定的请求路径或者方法进行处理。
  • 坏处:拦截器主要用于对请求的预处理和后处理,不够直接地集中在参数填充的功能上,可能需要额外的配置和管理。

总结

选择使用AOP来实现参数重写,是为了提高代码的复用性、可维护性和灵活性。它能够有效地解耦业务逻辑和横切关注点(如参数填充),使得代码更加清晰和易于扩展。然而,AOP也不是万能的解决方案,需要权衡其引入的复杂性和可能的运行时开销。

除了AOP,还可以考虑使用Controller基类方法或者拦截器来实现类似的功能,具体选择取决于项目的需求和团队的技术栈。

实现

下面是一个改进的示例,演示如何使用反射来处理不同类型的实体对象:

@Aspect
@Component
public class ControllerAspect {

    @Autowired
    private UserService userService; // 假设需要从userService中获取当前用户信息

    @Around("execution(* com.example.controller.*Controller.save*(..))")
    public Object aroundSave(ProceedingJoinPoint joinPoint) throws Throwable {
        // 获取目标方法的参数
        Object[] args = joinPoint.getArgs();

        for (Object arg : args) {
            if (arg instanceof BaseEntity) {
                handleEntity((BaseEntity) arg);
            }
        }

        // 继续执行目标方法
        Object result = joinPoint.proceed();

        return result;
    }

    private void handleEntity(BaseEntity entity) {
        try {
            // 使用反射获取实体对象的类
            Class<?> clazz = entity.getClass();

            // 设置创建时间、修改时间、创建人、修改人等通用属性
            Field createTimeField = clazz.getDeclaredField("createTime");
            createTimeField.setAccessible(true);
            createTimeField.set(entity, LocalDateTime.now());

            Field updateTimeField = clazz.getDeclaredField("updateTime");
            updateTimeField.setAccessible(true);
            updateTimeField.set(entity, LocalDateTime.now());

            Field createUserField = clazz.getDeclaredField("createUser");
            createUserField.setAccessible(true);
            createUserField.set(entity, userService.getCurrentUser());

            Field updateUserField = clazz.getDeclaredField("updateUser");
            updateUserField.setAccessible(true);
            updateUserField.set(entity, userService.getCurrentUser());

            // 可以根据需要添加其他通用属性的处理
        } catch (NoSuchFieldException | IllegalAccessException e) {
            // 处理异常
            e.printStackTrace();
        }
    }
}

解释说明:

  1. @Around("execution( com.example.controller.Controller.save(…))")*:

    • @Around:表示在目标方法执行前后都会执行该切面逻辑。
    • "execution(* com.example.controller.*Controller.save*(..))":指定切入点表达式,匹配所有保存方法(如save、saveOrUpdate等)。
  2. aroundSave方法

    • ProceedingJoinPoint joinPoint:继承自JoinPoint,可以控制目标方法的执行。
    • Object[] args = joinPoint.getArgs():获取目标方法的所有参数。
    • 遍历参数数组,对每个参数进行类型判断。在示例中,假设所有的实体类都继承自BaseEntity,因此使用instanceof BaseEntity来判断。
  3. handleEntity方法

    • BaseEntity entity:传入的实体对象,通过反射动态设置通用属性。
    • 使用entity.getClass()获取实体对象的Class对象,然后使用反射操作这些属性。
    • 获取并设置实体对象的创建时间、修改时间、创建人、修改人等通用属性。
    • 可以根据实际需求,添加其他通用属性的处理。

通过这种方式,你可以处理多种不同类型的实体对象,只需在BaseEntity中定义通用的属性,并确保这些属性在各个实体对象中都存在和可访问。这种实现方式不仅通用,而且具有较高的灵活性和可扩展性,能够满足处理复杂业务场景下多样化实体对象的需求。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/843314.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2-41 基于matlab的小车倒立摆系统的控制及GUI动画演示

基于matlab的小车倒立摆系统的控制及GUI动画演示。输入小车及倒立摆的初始参数&#xff0c;位置参数&#xff0c;对仿真时间和步长进行设置&#xff0c;通过LQR计算K值&#xff0c;进行角度、角速度、位置、速度仿真及曲线输出&#xff0c;程序已调通&#xff0c;可直接运行。 …

昇思MindSpore学习总结十七 —— 基于MindSpore通过GPT实现情感分类

1、要求 2、导入了一些必要的库和模块 以便在使用MindSpore和MindNLP进行深度学习任务时能使用各种功能&#xff0c;比如数据集处理、模型训练、评估和回调功能。 import os # 导入操作系统相关功能的模块&#xff0c;如文件和目录操作import mindspore # 导入MindSpore库&a…

HTTPServer改进思路2(mudou库核心思想融入)

mudou网络库思想理解 Reactor与多线程 服务器构建过程中&#xff0c;不仅仅使用一个Reactor&#xff0c;而是使用多个Reactor&#xff0c;每个Reactor执行自己专属的任务&#xff0c;从而提高响应效率。 首先Reactor是一种事件驱动处理模式&#xff0c;其主要通过IO多路复用…

基于WebGoat平台的SQL注入攻击

目录 引言 一、安装好JAVA 二、下载并运行WebGoat 三、注册并登录WebGoat 四、模拟攻击 1. 第九题 2. 第十题 3. 第十一题 4. 第十二题 5. 第十三题 五、思考体会 1. 举例说明SQL 注入攻击发生的原因。 2. 从信息的CIA 三要素&#xff08;机密性、完整性、可用性&…

JAVA:Filer过滤器+案例:请求IP访问限制和请求返回值修改

JAVA&#xff1a;Filer过滤器 介绍 Java中的Filter也被称为过滤器&#xff0c;它是Servlet技术的一部分&#xff0c;用于在web服务器上拦截请求和响应&#xff0c;以检查或转换其内容。 Filter的urlPatterns可以过滤特定地址http的请求&#xff0c;也可以利用Filter对访问请求…

[数据分析]脑图像处理工具

###############ATTENTION&#xff01;############### 非常需要注意软件适配的操作系统&#xff01;有些仅适用于Linux&#xff0c;可以点进各自软件手册查看详情。 需要自行查看支持的影像模态。 代码库和软件我没有加以区分。 不是专门预处理的博客&#xff01;&#xf…

Richteck立锜科技电源管理芯片简介及器件选择指南

一、电源管理简介 电源管理组件的选择和应用本身的电源输入和输出条件是高度关联的。 输入电源是交流或直流&#xff1f;需求的输出电压比输入电压高或是低&#xff1f;负载电流多大&#xff1f;系统是否对噪讯非常敏感&#xff1f;也许系统需要的是恒流而不是稳压 (例如 LED…

Mac装虚拟机占内存吗 Mac用虚拟机装Windows流畅吗

如今&#xff0c;越来越多的Mac用户选择在他们的设备上安装虚拟机来运行不同的操作系统。其中&#xff0c;最常见的是使用虚拟机在Mac上运行Windows。然而&#xff0c;许多人担心在Mac上装虚拟机会占用大量内存&#xff0c;影响电脑系统性能。此外&#xff0c;有些用户还关心在…

k8s中部署nacos

1 部署nfs # 在k8s的主节点上执行 mkdir -p /appdata/download cd /appdata/download git clone https://github.com/nacos-group/nacos-k8s.git 将nacos部署到middleware的命名空间中 kubectl create namespace middleware cd /appdata/download/nacos-k8s # 创建角色 kub…

图论模型-迪杰斯特拉算法和贝尔曼福特算法★★★★

该博客为个人学习清风建模的学习笔记&#xff0c;部分课程可以在B站&#xff1a;【强烈推荐】清风&#xff1a;数学建模算法、编程和写作培训的视频课程以及Matlab等软件教学_哔哩哔哩_bilibili 目录 ​1图论基础 1.1概念 1.2在线绘图 1.2.1网站 1.2.2MATLAB 1.3无向图的…

ABAP打印WORD的解决方案

客户要求按照固定格式输出到WORD模板中&#xff0c;目前OLE和DOI研究了均不太适合用于这种需求。 cl_docx_document类可以将WORD转化为XML文件&#xff0c;利用替换字符串方法将文档内容进行填充同 时不破坏WORD现有格式。 首先需要将WORD的单元格用各种预定义的字符进行填充…

canvas:矢量点转栅格

案例描述 ArcGIS提供了“点转栅格”的工具,可以将矢量点转换为栅格数据,以下尝试基于canvas绘图技术,实现经纬度矢量点转换为canvas栅格数据,并在Cesium.js三维地图中进行渲染。 原始数据 转出栅格 案例分析 实现的关键点在于:如何将经纬度坐标与canvas画布坐标进…

【Vue3】工程创建及目录说明

【Vue3】工程创建及目录说明 背景简介开发环境开发步骤及源码 背景 随着年龄的增长&#xff0c;很多曾经烂熟于心的技术原理已被岁月摩擦得愈发模糊起来&#xff0c;技术出身的人总是很难放下一些执念&#xff0c;遂将这些知识整理成文&#xff0c;以纪念曾经努力学习奋斗的日…

Figma 中文版指南:获取和安装汉化插件

Figma是一种主流的在线团队合作设计工具&#xff0c;也是一种基于 Web 端的设计工具。在当今的设计时代&#xff0c;Figma 的使用满足了每个人的设计需求&#xff0c;不仅可以实现在线编辑&#xff0c;还可以方便日常管理&#xff0c;有效提高工作效率。然而&#xff0c;相信很…

Java查询ES报错 I/O 异常解决方法: Request cannot be executed; I/O reactor status: STOPPED

问题 ES Request cannot be executed; I/O reactor status: STOPPED 报错解决 在使用ES和SpringBoot进行数据检索时&#xff0c;在接口中第一次搜索正常。第二次在搜索时在控制台就会输出Request cannot be executed; I/O reactor status: STOPPED错误 原因 本文错误是因为在使…

51单片机14(独立按键实验)

一、按键介绍 1、按键是一种电子开关&#xff0c;使用的时候&#xff0c;只要轻轻的按下我们的这个按钮&#xff0c;按钮就可以使这个开关导通。 2、当松开这个手的时候&#xff0c;我们的这个开关&#xff0c;就断开开发板上使用的这个按键&#xff0c;它的内部结构&#xff…

用Java手写jvm之实现java -version的效果

写在前面 源码 。 本文来用纯纯的Java代码来实现java -version的效果&#xff0c;就像下面这样&#xff1a; 1&#xff1a;程序 这里输出类似这样的&#xff1a; java version "9" Java(TM) SE Runtime Environment (build 9181) Java HotSpot(TM) 64-Bit Serve…

突破•指针二

听说这是目录哦 复习review❤️野指针&#x1fae7;assert断言&#x1fae7;assert的神奇之处 指针的使用和传址调用&#x1fae7;数组名的理解&#x1fae7;理解整个数组和数组首元素地址的区别 使用指针访问数组&#x1fae7;一维数组传参的本质&#x1fae7;二级指针&#x…

filebeat,kafka,clickhouse,ClickVisual搭建轻量级日志平台

springboot集成链路追踪 springboot版本 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.6.3</version><relativePath/> <!-- lookup parent from…

python—爬虫爬取电影页面实例

下面是一个简单的爬虫实例&#xff0c;使用Python的requests库来发送HTTP请求&#xff0c;并使用lxml库来解析HTML页面内容。这个爬虫的目标是抓取一个电影网站&#xff0c;并提取每部电影的主义部分。 首先&#xff0c;确保你已经安装了requests和lxml库。如果没有安装&#x…