Spring AOP -- 面相切面编程

简介: 【2月更文挑战第1天】基于注解 @Aspect;切点表达式;通知类型;多种通知类型的执行顺序;@Pointcut;多个切面类的执行顺序;@Order;基于自定义注解;给已有的注解进行功能的增强;AOP的优势:代码无侵入:不修改原始的业务方法,就可以对原始的业务方法进行了功能的增强或者是功能的改变;减少了重复代码;提高开发效率;维护方便。​

AOP是Spring框架的核心之一,AOP是一种思想,它的实现方法有很多,有Spring AOP,也有AspectJ、CGLIB等。我们熟知的拦截器其实就是AOP思想的一种实现方式。

AOP是一种思想,是对某一类事情的集中处理。

Spring AOP的实现方式:

  1. 基于注解 @Aspect;
  2. 基于自定义注解;
  3. 基于Spring API(通过xml配置的方式);
  4. 基于代理来实现。

想要实现Spring Aop需要先引入以下依赖。

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

image.gif

例如:我们此时想要优化一个接口的执行效率

此时有一个接口如下:

@RequestMapping("/aop")
@RestController
public class Main {
    @Autowired
    private ForService fs;
    @RequestMapping("/fun1")
    public void fun1() {
        fs.fun(3);
        for (int i = 0; i < 1000; i++) {};
    }
}

image.gif

@Service
public class ForService {
    public int fun(int i) {
        for (int j = 0; j < 3000; j++) {
            i++;
        }
        return i;
    }
}

image.gif

我们首先需要知道这个接口在执行过程中调用的各个方法的执行时间,然后再对每个方法进行针对性优化。用AOP思想来实现:

基于注解 @Aspect

@Slf4j
@Component
@Aspect
public class WritTime {
    @Around("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public Object time(ProceedingJoinPoint pjp) throws Throwable {
        //记录开始时间
        long start = System.currentTimeMillis();
        //执行目标方法
        Object a = pjp.proceed();
        //打印方法执行时间。pjp.toShortString()方法会返回方法签名的简写
        log.info(pjp.toShortString()+":"+(System.currentTimeMillis()-start)+"ms");
        return a;
    }
}

image.gif

image.gif

image.gif

  • 切点:也称之为"切入点",提供⼀组规则(切点表达式)告诉程序对哪些方法来进行功能增强;
  • 连接点:满足切点表达式规则的方法,就是连接点。也就是可以被AOP控制的方法;
  • 通知:就是具体要做的工作,指哪些重复的逻辑,也就是共性功能(最终体现为⼀个方法);
  • 切面:切点+通知。
  • 切面类:切面所在的类,一个切面类中可以包含多个切面。

切点表达式:

常见的切点表达式有两种表达方式:

  1. execution(……):根据方法的签名来匹配
  2. @annotation(……):根据注解匹配

image.gif

访问修饰限定符和异常可以被省略。

* :匹配任意字符,只匹配一个元素(返回类型,包,类名,方法或者方法参数)

  • 包名使用 * 表示任意包(一层包使用一个*);
  • 类名使用 * 表示任意类;
  • 返回值使用 * 表示任意返回值类型;
  • 方法名使用 * 表示任意方法;
  • 参数使用 * 表示⼀个任意类型的参数。

.. :匹配多个连续的任意符号,可以通配任意层级的包,或任意类型,任意个数的参数

  • 使用 .. 配置包名,标识此包以及此包下的所有子包;
  • 可以使用 .. 配置参数,任意个任意类型的参数。

+:表示匹配当前类和其子类

通知类型:

  • @Around:环绕通知,此注解标注的通知方法在目标方法前后都被执行;
  • @Before:前置通知,此注解标注的通知方法在目标方法前被执行;
  • @After:后置通知,此注解标注的通知方法在目标方法后被执行;
  • @AfterReturning:返回后通知;
  • @AfterThrowing:异常后通知。

多种通知类型的执行顺序

现有如下接口:

@Slf4j
@RequestMapping("/aop")
@RestController
public class Main {
    @RequestMapping("/fun1")
    public void fun1() {
        log.info("执行fun1方法");
    }
}

image.gif

定义如下切面类,里面有多个切面。

@Slf4j
@Component
@Aspect
public class WritTime {
    @Around("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public Object time(ProceedingJoinPoint pjp) throws Throwable {
        log.info("@Around 方法前");
        //执行目标方法
        Object a = pjp.proceed();
        log.info("@Around 方法后");
        return a;
    }
    @Before("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void before() {
        log.info("@Before 前置通知");
    }
    @After("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void After() {
        log.info("@After 后置通知");
    }
    @AfterReturning("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void AfterReturning() {
        log.info("@AfterReturning 返回后通知");
    }
    @AfterThrowing("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void AfterThrowing() {
        log.info("@AfterThrowing 异常后通知");
    }
}

image.gif

image.gif

如果发生异常:

image.gif

异常是可以再切面中被捕获的,但需要通知类型为@Around

image.gif

@Pointcut

@Pointcut注解可以把公共的切点表达式提取出来,需要用到时引用该切入点表达式即可。

@Slf4j
@Component
@Aspect
public class WritTime {
    @Pointcut("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void pc(){}
    @Around("pc()")
    public Object time(ProceedingJoinPoint pjp) {
        log.info("@Around 方法前");
        //执行目标方法
        Object a = null;
        try {
            a = pjp.proceed();
        } catch (Throwable e) {
        }
        log.info("@Around 方法后");
        return a;
    }
    @Before("pc()")
    public void before() {
        log.info("@Before 前置通知");
    }
    @After("pc()")
    public void After() {
        log.info("@After 后置通知");
    }
    @AfterReturning("pc()")
    public void AfterReturning() {
        log.info("@AfterReturning 返回后通知");
    }
    @AfterThrowing("pc()")
    public void AfterThrowing() {
        log.info("@AfterThrowing 异常后通知");
    }
}

image.gif

image.gif

声明的切面表达式也可以再其他切面类中使用但需要提前声明

@Slf4j
@Component
@Aspect
public class WritTime1 {
    //()里引用的切面表达式前必须加上全限定类名
    @Before("com.example.Spring_demo.aop.WritTime.pc()")
    public void before() {
        log.info("WritTime1通知");
    }
}

image.gif

image.gif

多个切面类的执行顺序

@Slf4j
@RequestMapping("/aop")
@RestController
public class Main {
    @RequestMapping("/fun1")
    public void fun1() {
        log.info("执行fun1方法");
    }
}

image.gif

定义以下三个切面类,每个类里面只有一个切面:

//第一个
@Slf4j
@Component
@Aspect
public class WritTime1 {
    @Before("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void before() {
        log.info("WritTime1通知");
    }
}
//第二个
@Slf4j
@Component
@Aspect
public class WritTime2 {
    @Before("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void before() {
        log.info("WritTime2通知");
    }
}
//第三个
@Slf4j
@Component
@Aspect
public class WritTime3 {
    @Before("execution(* com.example.Spring_demo.aop.*.*(..)))")
    public void before() {
        log.info("WritTime3通知");
    }
}

image.gif

image.gif

如果存在多个切面类执行顺序默认是按照类名进行排序。

@Order

当存在多个切面类时可以通过@Order注解来定义各个切面类的优先级。

对上述切面类进行细微修改加上注解

//第一个
@Slf4j
@Component
@Aspect
@Order(3)
public class WritTime1 {
    ……
}
//第二个
@Slf4j
@Component
@Aspect
@Order(2)
public class WritTime2 {
    ……
}
//第三个
@Slf4j
@Component
@Aspect
@Order(1)
public class WritTime3 {
    ……
}

image.gif

image.gif

基于自定义注解

使用自定义注解来计算方法的执行时间。

声明一个自定义注解:

//表示该接口只能作用于方法
@Target({ElementType.METHOD})
//该接口的生命周期
@Retention(RetentionPolicy.RUNTIME)
public @interface Time {
}

image.gif

使用切面类实现该注解的功能:

@Slf4j
@Component
@Aspect
public class WritTime {
    //@annotation里面的值为要实现的目标注解的全限定类名
    @Around("@annotation(com.example.Spring_demo.aop.Time)")
    public Object time(ProceedingJoinPoint pjp) throws Throwable {
        //记录开始时间
        long start = System.currentTimeMillis();
        //执行目标方法
        Object a = pjp.proceed();
        //打印方法执行时间。pjp.toShortString()方法会返回方法签名的简写
        log.info(pjp.toShortString()+":"+(System.currentTimeMillis()-start)+"ms(注解实现)");
        return a;
    }
}

image.gif

给待测方法添加注解

@Slf4j
@RequestMapping("/aop")
@RestController
public class Main {
    @Autowired
    private ForService forService;
    @Time
    @RequestMapping("/fun1")
    public void fun1() {
        forService.fun(1);
        for (int i = 0; i < 10; i++) {}
    }
}

image.gif

image.gif

给已有的注解进行功能的增强

还可以使用上面的方式给已有的注解进行功能的增强(给@RequestMapping注解添加可以打印接口执行时间的功能):

@Slf4j
@Component
@Aspect
public class WritTime {
    //@annotation里面的值为@RequestMapping注解的全限定类名   
    @Around("@annotation(org.springframework.web.bind.annotation.RequestMapping)")
    public Object time(ProceedingJoinPoint pjp) throws Throwable {
        //记录开始时间
        long start = System.currentTimeMillis();
        //执行目标方法
        Object a = pjp.proceed();
        //打印方法执行时间。pjp.toShortString()方法会返回方法签名的简写
        log.info(pjp.toShortString()+":"+(System.currentTimeMillis()-start)+"ms(注解实现)");
        return a;
    }
}

image.gif

image.gif

AOP的优势:

  1. 代码无侵入:不修改原始的业务方法,就可以对原始的业务方法进行了功能的增强或者是功能的改变;
  2. 减少了重复代码;
  3. 提高开发效率;
  4. 维护方便。
目录
相关文章
|
4月前
|
XML 安全 Java
使用 Spring 的 @Aspect 和 @Pointcut 注解简化面向方面的编程 (AOP)
面向方面编程(AOP)通过分离横切关注点,如日志、安全和事务,提升代码模块化与可维护性。Spring 提供了对 AOP 的强大支持,核心注解 `@Aspect` 和 `@Pointcut` 使得定义切面与切入点变得简洁直观。`@Aspect` 标记切面类,集中处理通用逻辑;`@Pointcut` 则通过表达式定义通知的应用位置,提高代码可读性与复用性。二者结合,使开发者能清晰划分业务逻辑与辅助功能,简化维护并提升系统灵活性。Spring AOP 借助代理机制实现运行时织入,与 Spring 容器无缝集成,支持依赖注入与声明式配置,是构建清晰、高内聚应用的理想选择。
565 0
|
3月前
|
XML Java 数据格式
《深入理解Spring》:AOP面向切面编程深度解析
Spring AOP通过代理模式实现面向切面编程,将日志、事务等横切关注点与业务逻辑分离。支持注解、XML和编程式配置,提供五种通知类型及丰富切点表达式,助力构建高内聚、低耦合的可维护系统。
|
8月前
|
监控 安全 Java
Spring AOP实现原理
本内容主要介绍了Spring AOP的核心概念、实现机制及代理生成流程。涵盖切面(Aspect)、连接点(Join Point)、通知(Advice)、切点(Pointcut)等关键概念,解析了JDK动态代理与CGLIB代理的原理及对比,并深入探讨了通知执行链路和责任链模式的应用。同时,详细分析了AspectJ注解驱动的AOP解析过程,包括切面识别、切点表达式匹配及通知适配为Advice的机制,帮助理解Spring AOP的工作原理与实现细节。
1305 13
|
5月前
|
人工智能 监控 安全
如何快速上手【Spring AOP】?核心应用实战(上篇)
哈喽大家好吖~欢迎来到Spring AOP系列教程的上篇 - 应用篇。在本篇,我们将专注于Spring AOP的实际应用,通过具体的代码示例和场景分析,帮助大家掌握AOP的使用方法和技巧。而在后续的下篇中,我们将深入探讨Spring AOP的实现原理和底层机制。 AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的核心特性之一,它能够帮助我们解决横切关注点(如日志记录、性能统计、安全控制、事务管理等)的问题,提高代码的模块化程度和复用性。
|
5月前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
|
11月前
|
XML Java 测试技术
Spring AOP—通知类型 和 切入点表达式 万字详解(通俗易懂)
Spring 第五节 AOP——切入点表达式 万字详解!
849 25
|
11月前
|
XML 安全 Java
Spring AOP—深入动态代理 万字详解(通俗易懂)
Spring 第四节 AOP——动态代理 万字详解!
497 24
|
10月前
|
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在实际项目中的重要作用,并提供了课程源码下载链接供进一步学习。
1389 0
|
10月前
|
Java 开发者 微服务
微服务——SpringBoot使用归纳——Spring Boot中的切面AOP处理——什么是AOP
本文介绍了Spring Boot中的切面AOP处理。AOP(Aspect Oriented Programming)即面向切面编程,其核心思想是分离关注点。通过AOP,程序可以将与业务逻辑无关的代码(如日志记录、事务管理等)从主要逻辑中抽离,交由专门的“仆人”处理,从而让开发者专注于核心任务。这种机制实现了模块间的灵活组合,使程序结构更加可配置、可扩展。文中以生活化比喻生动阐释了AOP的工作原理及其优势。
550 0