五大通知类型
1、前置通知:在目标方法执行之前执行执行的通知
前置通知方法,可以没有参数,也可以额外接收一个JoinPoint,Spring会自动将该对象传入,代表当前的连接点,通过该对象可以获取目标对象和目标方法相关的信息。
注意,如果接收JoinPoint,必须保证其为方法的第一个参数,否则报错。
2、环绕通知:目标方法执行之前和之后都可以执行额外代码的通知
在环绕通知中必须手动的调用目标方法,否则目标方法不会执行。这个手动调用是通过ProceedingJoinPoint来实现的,可以在环绕通知中接收一个此类型的形参,spring容器会自动将该对象传入,这个参数必须处在环绕通知的第一个形参位置。
ProceedingJoinPoint时JoinPoint的子类,要注意,只有环绕通知可以接收ProceedingJoinPoint,而其他通知只能接收JoinPoint。
配置方式:环绕通知需要手动返回返回值,否则真正调用者将拿不到返回值,只能得到一个null。
环绕通知有:控制目标方法是否执行;目标方法执行之前或之后执行额外代码;控制是否返回返回值;改变返回值的能力,环绕通知虽然有这样的能力,但一定要慎用,要小心不要破坏了软件分层的“高内聚 低耦合”的目标。
3、后置通知:在目标方成功执行之后执行的通知。
在后置通知中也可以选择性的接收一个JoinPoint来获取连接点的额外信息,但是这个参数必须处在参数列表的第一个。
4、异常通知:在目标方法抛出异常时执行的通知。
可以配置传入JoinPoint获取目标对象和目标方法相关信息,但必须处在参数列表第一位。另外,还可以配置参数,让异常通知接收到目标方法抛出的异常对象
5、最终通知:在目标方法执行之后执行的通知。
和后置通知不同之处在于,后置通知是在方法正常返回后执行的通知,如果方法没有正常返,例如抛出异常,则后置通知不会执行。而最终通知无论如何都会在目标方法调用过后执行,即使目标方法没有正常的执行完成。
另外,后置通知可以通过配置得到返回值,而最终通知无法得到。
最终通知也可以额外接收一个JoinPoint参数,来获取目标对象和目标方法相关信息,但一定要保证必须是第一个参数。
Spring5 中Aop五种通知执行的顺序
a. 在目标方法没有抛出异常的情况下:
环绕前置通知处理
@Before前置通知
invokeMethod()
@AfterReturning返回后通知:执行方法结束前执行
@After后置通知
环绕后置通知处理
b.在目标方法抛出异常的情况下:
环绕前置通知处理
@Before前置通知
invokeMethod()
@AfterThrowing异常通知:出现异常时执行
@After后置通知
c.如果存在多个切面:
多切面执行时,采用了责任链设计模式。
切面的配置顺序决定了切面的执行顺序,多个切面执行的过程,类似于方法调用的过程,在环绕通知的proceed()执行时,去执行下一个切面或如果没有下一个切面执行目标方法。
五种通知常见使用场景
前置通知:记录日志(方法被调用)
环绕通知:控制事务、权限控制
后置通知:记录日志(方法已成功调用)
异常通知:异常处理、控制事务
最终通知:记录日志(方法已调用,但不一定成功)
AOP
1.1 AOP简介和作用
简介:AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,指导开发者如何组织程序结构。
作用:在不惊动原始设计的基础上为其进行功能增强。简单的说就是在不改变方法源代码的基础上对方法进行功能增强。
1.1.1 AOP相关依赖
<dependencies>
<!-- Spring 核心 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<!--spring核心依赖,会将spring-aop传递进来-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.2.10.RELEASE</version>
</dependency>
<!--切入点表达式依赖,目的是找到切入点方法,也就是找到要增强的方法-->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>
</dependencies>
1.1.2 AOP相关注解
@Aspect 注解 使之成为切面类
@Pointcut("execution(public * com.MultipleDataSource.service.impl.*.*(..))") 使在用构造函数上面; 使之对部分类进行切点;
@Before("webLog()") 参数为构造方法 即对象构造之前执行的方法;
@AfterReturning(returning = "ret", pointcut = "webLog()") 参数为构造方法 即对象构造之前执行的方法;参数returning = "ret"获取方法执行结果值;ret是变量名接收返回值。