AOP概述
- AOP:Aspect Oriented Programming(面向切面编程、面向方面编程),其实就是面向特定方法编程。
- 面向切面编程(AOP)是一种编程范式,它允许开发者在不修改业务逻辑代码的前提下,对程序中的多个位置进行统一的、横切的处理。AOP 的核心概念包括切面(Aspect)、连接点(Joinpoint)、通知(Advice)、切入点(Pointcut)和织入(Weaving)。
- AOP 的应用场景非常广泛,包括但不限于日志记录、事务管理、安全性控制、性能监控等方面。它通过将横切关注点与业务逻辑分离,提高了代码的可维护性和可重用性。在 Spring 中,可以使用 @Aspect 注解来定义一个切面,并通过 @Pointcut 来指定切点,使用 @Before、@After、@Around 等注解来定义通知。
AOP核心概念
- 连接点:JoinPoint,可以被AOP控制的方法(暗含方法执行时的相关信息)
- 通知:Advice,指哪些重复的逻辑,也就是共性功能(最终体现为一个方法)
- 切入点:PointCut,匹配连接点的条件,通知仅会在切入点方法执行时被应用
- 切面:Aspect,描述通知与切入点的对应关系(通知+切入点)
- 目标对象:Target,通知所应用的对象
下面是图片所示:
AOP执行流程
SpringAOP底层是基于动态代理技术来实现的,在程序运行的时候会自动为目标对象生成一个动态代理对象,这个动态代理对象就会对原始的方法进行功能的增强,在目标方法之前先进行记录当前的系统时间,然后在运行原始的方法,最后在记录方法的结束时间,最后再来统计方法的耗时,此时代理对象就已经对目标对象的方法进行增强,而在DeptController类里面注入的deptService其实是代理对象而不是目标对象,在调用list()方法的时候其实调用的是代理对象中的list()方法·。
通知类型
- @Around:环绕通知,此注解标注的通知方法在目标方法前、后都被执行
- @Before:前置通知,此注解标注的通知方法在目标方法前被执行
- @After :后置通知,此注解标注的通知方法在目标方法后被执行,无论是否有异常都会执行
- @AfterReturning :返回后通知,此注解标注的通知方法在目标方法后被执行,有异常不会执行
- @AfterThrowing: 异常后通知,此注解标注的通知方法发生异常后执行
注意事项:
- Around环绕通知需要自己调用ProceedingJoinPoint.proceed()来让原始方法执行,其他通知不需要考虑目标方法执行
- @Around环绕通知方法的返回值,必须指定为0bject,来接收原始方法的返回值。
@Pointcut
- 该注解的作用是将公共的切点表达式抽取出来,需要用到时引用该切点表达式即可
通知顺序
当有多个切面的切入点都匹配到了目标方法,目标方法运行时,多个通知方法都会被执行.
执行顺序
1.不同切面类中,默认按照切面类的类名字母排序
- 目标方法前的通知方法:字母排名靠前的先执行
- 目标方法后的通知方法:字母排名靠前的后执行
2.用 @Order(数字) 加在切面类上来控制顺序
- 目标方法前的通知方法:数字小的先执行
- 目标方法后的通知方法:数字小的后执行
切入点表达式
- 切入点表达式:描述切入点方法的一种表达方式
- 作用:主要用来决定项目中的哪些方法需要加入通知
常见形式:
1.execution(…):根据方法的签名来匹配
2.@annotation(…):根据注解匹配
切入点表达式-execution
execution 主要根据方法的返回值、包名、类名、方法名、方法参数等信息来匹配,语法为:
其中带 ?的表示可以省略的部分
- 访问修饰符:可省略(比如:public、protected)
- 包名.类名: 可省略
- throws 异常:可省略(注意是方法上声明抛出的异常,不是实际抛出的异常)
可以使用通配符描述切入点
- *:单个独立的任意符号,可以通配任意返回值、包名、类名、方法名、任意类型的一个参数,也可以通配包、类、方法名的一部分
- …:多个连续的任意符号,可以通配任意层级的包,或任意类型、任意个数的参数
注意事项:根据业务需要,可以使用且(&&)、或(||)、非(!)来组合比较复杂的切入点表达式。
书写建议
- 所有业务方法名在命名时尽量规范,方便切入点表达式快速匹配。如:査询类方法都是 find 开头,更新类方法都是 update开头
- 描述切入点方法通常基于接口描述,而不是直接描述实现类,增强拓展性。
- 在满足业务需要的前提下,尽量缩小切入点的匹配范围。如:包名匹配尽量不使用…,使用*匹配单个包,
切入点表达式-annotation
@annotation切入点表达式,用于匹配标识有特定注解的方法
连接点
在Spring中用JoinPoint抽象了连接点,用它可以获得方法执行时的相关信息,如目标类名、方法名、方法参数等
-
对于 @Around 通知,获取连接点信息只能使用 ProceedingJoinPoint
-
对于其他四种通知,获取连接点信息只能使用 JoinPoint,它是 ProceedingJoinPoint的父类型