Spring系列六:AOP概念和使用

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: ​ AOP19.说说什么是AOP?AOP:面向切面编程。简单说,就是把一些业务逻辑中的相同的代码抽取到一个独立的模块中,让业务逻辑更加清爽。横向抽取具体来说,假如我现在要crud写一堆业务,可是如何业务代码前后前后进行打印日志和参数的校验呢?我们可以把日志记录和数据校验可重用的功能模块分离出来,然后在程序的执行的合适的地方动态地植入这些代码并执行。这样就简化了代码的书写。AOP应用示例业务逻辑代码中没有参和通用逻辑的代码,业务模块更简洁,只包含核心业务代码。实现了业务逻辑和通用逻辑的代码分离,便于维护和升级,降低了业务逻辑和通用逻辑的耦合性。AOP 可以将遍布

  AOP

19.说说什么是AOP?

AOP:面向切面编程。简单说,就是把一些业务逻辑中的相同的代码抽取到一个独立的模块中,让业务逻辑更加清爽。

image.gif编辑

横向抽取

具体来说,假如我现在要crud写一堆业务,可是如何业务代码前后前后进行打印日志和参数的校验呢?

我们可以把日志记录数据校验可重用的功能模块分离出来,然后在程序的执行的合适的地方动态地植入这些代码并执行。这样就简化了代码的书写。

image.gif编辑

AOP应用示例

业务逻辑代码中没有参和通用逻辑的代码,业务模块更简洁,只包含核心业务代码。实现了业务逻辑和通用逻辑的代码分离,便于维护和升级,降低了业务逻辑和通用逻辑的耦合性。

AOP 可以将遍布应用各处的功能分离出来形成可重用的组件。在编译期间、装载期间或运行期间实现在不修改源代码的情况下给程序动态添加功能。从而实现对业务逻辑的隔离,提高代码的模块化能力。

image.gif编辑

Java语言执行过程

AOP 的核心其实就是动态代理,如果是实现了接口的话就会使用 JDK 动态代理,否则使用 CGLIB 代理,主要应用于处理一些具有横切性质的系统级服务,如日志收集、事务管理、安全检查、缓存、对象池管理等。

AOP有哪些核心概念?

    • 切面(Aspect):类是对物体特征的抽象,切面就是对横切关注点的抽象
    • 连接点(Joinpoint):被拦截到的点,因为 Spring 只支持方法类型的连接点,所以在 Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
    • 切点(Pointcut):对连接点进行拦截的定位
    • 通知(Advice):所谓通知指的就是指拦截到连接点之后要执行的代码,也可以称作增强
    • 目标对象 (Target):代理的目标对象
    • 织入(Weabing):织入是将增强添加到目标类的具体连接点上的过程。
      • 编译期织入:切面在目标类编译时被织入
      • 类加载期织入:切面在目标类加载到JVM时被织入。需要特殊的类加载器,它可以在目标类被引入应用之前增强该目标类的字节码。
      • 运行期织入:切面在应用运行的某个时刻被织入。一般情况下,在织入切面时,AOP容器会为目标对象动态地创建一个代理对象。SpringAOP就是以这种方式织入切面。
        Spring采用运行期织入,而AspectJ采用编译期织入和类加载器织入。
        • 引介(introduction):引介是一种特殊的增强,可以动态地为类添加一些属性和方法

        AOP有哪些环绕方式?

        AOP 一般有 5 种环绕方式:

          • 前置通知 (@Before)
          • 返回通知 (@AfterReturning)
          • 异常通知 (@AfterThrowing)
          • 后置通知 (@After)
          • 环绕通知 (@Around)

          image.gif编辑

          环绕方式

          多个切面的情况下,可以通过 @Order 指定先后顺序,数字越小,优先级越高。

          20.说说你平时有用到AOP吗?

          PS:这道题老三的同事面试候选人的时候问到了,候选人说了一堆AOP原理,同事就势来一句,你能现场写一下AOP的应用吗?结果——场面一度很尴尬。虽然我对面试写这种百度就能出来的东西持保留意见,但是还是加上了这一问,毕竟招人最后都是要撸代码的。

          这里给出一个小例子,SpringBoot项目中,利用AOP打印接口的入参和出参日志,以及执行时间,还是比较快捷的。

            • 引入依赖:引入AOP依赖
            <dependency>
                        <groupId>org.springframework.boot</groupId>
                        <artifactId>spring-boot-starter-aop</artifactId>
                    </dependency>
            • image.gif
            • 自定义注解:自定义一个注解作为切点
            @Retention(RetentionPolicy.RUNTIME)
            @Target({ElementType.METHOD})
            @Documented
            public @interface WebLog {
            }
            • image.gif
            • 配置AOP切面:
            @Aspect
            @Component
            public class WebLogAspect {
                private final static Logger logger         = LoggerFactory.getLogger(WebLogAspect.class);
                /**
                 * 以自定义 @WebLog 注解为切点
                 **/
                @Pointcut("@annotation(cn.fighter3.spring.aop_demo.WebLog)")
                public void webLog() {}
                /**
                 * 在切点之前织入
                 */
                @Before("webLog()")
                public void doBefore(JoinPoint joinPoint) throws Throwable {
                    // 开始打印请求日志
                    ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
                    HttpServletRequest request = attributes.getRequest();
                    // 打印请求相关参数
                    logger.info("========================================== Start ==========================================");
                    // 打印请求 url
                    logger.info("URL            : {}", request.getRequestURL().toString());
                    // 打印 Http method
                    logger.info("HTTP Method    : {}", request.getMethod());
                    // 打印调用 controller 的全路径以及执行方法
                    logger.info("Class Method   : {}.{}", joinPoint.getSignature().getDeclaringTypeName(), joinPoint.getSignature().getName());
                    // 打印请求的 IP
                    logger.info("IP             : {}", request.getRemoteAddr());
                    // 打印请求入参
                    logger.info("Request Args   : {}",new ObjectMapper().writeValueAsString(joinPoint.getArgs()));
                }
                /**
                 * 在切点之后织入
                 * @throws Throwable
                 */
                @After("webLog()")
                public void doAfter() throws Throwable {
                    // 结束后打个分隔线,方便查看
                    logger.info("=========================================== End ===========================================");
                }
                /**
                 * 环绕
                 */
                @Around("webLog()")
                public Object doAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
                    //开始时间
                    long startTime = System.currentTimeMillis();
                    Object result = proceedingJoinPoint.proceed();
                    // 打印出参
                    logger.info("Response Args  : {}", new ObjectMapper().writeValueAsString(result));
                    // 执行耗时
                    logger.info("Time-Consuming : {} ms", System.currentTimeMillis() - startTime);
                    return result;
                }
            }
            • image.gif
              • @Aspect:标识切面
              • @Pointcut:设置切点,这里以自定义注解为切点,定义切点有很多其它种方式,自定义注解是比较常用的一种。
              • @Before:在切点之前织入,打印了一些入参信息
              • @Around:环绕切点,打印返回参数和接口执行时间
                • 使用:只需要在接口上加上自定义注解
                @GetMapping("/hello")
                    @WebLog(desc = "这是一个欢迎接口")
                    public String hello(String name){
                        return "Hello "+name;
                    }
                • image.gif
                • 执行结果:可以看到日志打印了入参、出参和执行时间


                  image.gif编辑


                相关实践学习
                通过日志服务实现云资源OSS的安全审计
                本实验介绍如何通过日志服务实现云资源OSS的安全审计。
                相关文章
                |
                4月前
                |
                监控 安全 Java
                Spring AOP实现原理
                本内容主要介绍了Spring AOP的核心概念、实现机制及代理生成流程。涵盖切面(Aspect)、连接点(Join Point)、通知(Advice)、切点(Pointcut)等关键概念,解析了JDK动态代理与CGLIB代理的原理及对比,并深入探讨了通知执行链路和责任链模式的应用。同时,详细分析了AspectJ注解驱动的AOP解析过程,包括切面识别、切点表达式匹配及通知适配为Advice的机制,帮助理解Spring AOP的工作原理与实现细节。
                |
                28天前
                |
                人工智能 监控 安全
                如何快速上手【Spring AOP】?核心应用实战(上篇)
                哈喽大家好吖~欢迎来到Spring AOP系列教程的上篇 - 应用篇。在本篇,我们将专注于Spring AOP的实际应用,通过具体的代码示例和场景分析,帮助大家掌握AOP的使用方法和技巧。而在后续的下篇中,我们将深入探讨Spring AOP的实现原理和底层机制。 AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的核心特性之一,它能够帮助我们解决横切关注点(如日志记录、性能统计、安全控制、事务管理等)的问题,提高代码的模块化程度和复用性。
                |
                26天前
                |
                设计模式 Java 开发者
                如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
                Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
                |
                8月前
                |
                XML Java 开发者
                Spring Boot中的AOP实现
                Spring AOP(面向切面编程)允许开发者在不修改原有业务逻辑的情况下增强功能,基于代理模式拦截和增强方法调用。Spring Boot通过集成Spring AOP和AspectJ简化了AOP的使用,只需添加依赖并定义切面类。关键概念包括切面、通知和切点。切面类使用`@Aspect`和`@Component`注解标注,通知定义切面行为,切点定义应用位置。Spring Boot自动检测并创建代理对象,支持JDK动态代理和CGLIB代理。通过源码分析可深入了解其实现细节,优化应用功能。
                389 6
                |
                7月前
                |
                XML Java 测试技术
                Spring AOP—通知类型 和 切入点表达式 万字详解(通俗易懂)
                Spring 第五节 AOP——切入点表达式 万字详解!
                349 25
                |
                7月前
                |
                XML 安全 Java
                Spring AOP—深入动态代理 万字详解(通俗易懂)
                Spring 第四节 AOP——动态代理 万字详解!
                251 24
                |
                6月前
                |
                Java API 微服务
                微服务——SpringBoot使用归纳——Spring Boot中的切面AOP处理——Spring Boot 中的 AOP 处理
                本文详细讲解了Spring Boot中的AOP(面向切面编程)处理方法。首先介绍如何引入AOP依赖,通过添加`spring-boot-starter-aop`实现。接着阐述了如何定义和实现AOP切面,包括常用注解如`@Aspect`、`@Pointcut`、`@Before`、`@After`、`@AfterReturning`和`@AfterThrowing`的使用场景与示例代码。通过这些注解,可以分别在方法执行前、后、返回时或抛出异常时插入自定义逻辑,从而实现功能增强或日志记录等操作。最后总结了AOP在实际项目中的重要作用,并提供了课程源码下载链接供进一步学习。
                687 0
                |
                6月前
                |
                Java 开发者 微服务
                微服务——SpringBoot使用归纳——Spring Boot中的切面AOP处理——什么是AOP
                本文介绍了Spring Boot中的切面AOP处理。AOP(Aspect Oriented Programming)即面向切面编程,其核心思想是分离关注点。通过AOP,程序可以将与业务逻辑无关的代码(如日志记录、事务管理等)从主要逻辑中抽离,交由专门的“仆人”处理,从而让开发者专注于核心任务。这种机制实现了模块间的灵活组合,使程序结构更加可配置、可扩展。文中以生活化比喻生动阐释了AOP的工作原理及其优势。
                294 0
                |
                8月前
                |
                XML Java 开发者
                Spring底层架构核心概念解析
                理解 Spring 框架的核心概念对于开发和维护 Spring 应用程序至关重要。IOC 和 AOP 是其两个关键特性,通过依赖注入和面向切面编程实现了高效的模块化和松耦合设计。Spring 容器管理着 Beans 的生命周期和配置,而核心模块为各种应用场景提供了丰富的功能支持。通过全面掌握这些核心概念,开发者可以更加高效地利用 Spring 框架开发企业级应用。
                233 18
                |
                8月前
                |
                存储 安全 Java
                Spring Boot 3 集成Spring AOP实现系统日志记录
                本文介绍了如何在Spring Boot 3中集成Spring AOP实现系统日志记录功能。通过定义`SysLog`注解和配置相应的AOP切面,可以在方法执行前后自动记录日志信息,包括操作的开始时间、结束时间、请求参数、返回结果、异常信息等,并将这些信息保存到数据库中。此外,还使用了`ThreadLocal`变量来存储每个线程独立的日志数据,确保线程安全。文中还展示了项目实战中的部分代码片段,以及基于Spring Boot 3 + Vue 3构建的快速开发框架的简介与内置功能列表。此框架结合了当前主流技术栈,提供了用户管理、权限控制、接口文档自动生成等多项实用特性。
                532 8