1、前置通知:在目标方法执行之前执行执行的通知
前置通知方法,可以没有参数,也可以额外接收一个JoinPoint,Spring会自动将该对象传入,代表当前的连接点,通过该对象可以获取目标对象和目标方法相关的信息。
注意,如果接收JoinPoint,必须保证其为方法的第一个参数,否则报错。
2、环绕通知:目标方法执行之前和之后都可以执行额外代码的通知
在环绕通知中必须手动的调用目标方法,否则目标方法不会执行。这个手动调用是通过ProceedingJoinPoint来实现的,可以在环绕通知中接收一个此类型的形参,spring容器会自动将该对象传入,这个参数必须处在环绕通知的第一个形参位置。
ProceedingJoinPoint时JoinPoint的子类,要注意,只有环绕通知可以接收ProceedingJoinPoint,而其他通知只能接收JoinPoint。
配置方式:环绕通知需要手动返回返回值,否则真正调用者将拿不到返回值,只能得到一个null。
环绕通知有:控制目标方法是否执行;目标方法执行之前或之后执行额外代码;控制是否返回返回值;改变返回值的能力,环绕通知虽然有这样的能力,但一定要慎用,要小心不要破坏了软件分层的“高内聚 低耦合”的目标。
3、后置通知:在目标方成功执行之后执行的通知。
在后置通知中也可以选择性的接收一个JoinPoint来获取连接点的额外信息,但是这个参数必须处在参数列表的第一个。
4、异常通知:在目标方法抛出异常时执行的通知。
可以配置传入JoinPoint获取目标对象和目标方法相关信息,但必须处在参数列表第一位。另外,还可以配置参数,让异常通知接收到目标方法抛出的异常对象
5、最终通知:在目标方法执行之后执行的通知。
和后置通知不同之处在于,后置通知是在方法正常返回后执行的通知,如果方法没有正常返,例如抛出异常,则后置通知不会执行。而最终通知无论如何都会在目标方法调用过后执行,即使目标方法没有正常的执行完成。
另外,后置通知可以通过配置得到返回值,而最终通知无法得到。
最终通知也可以额外接收一个JoinPoint参数,来获取目标对象和目标方法相关信息,但一定要保证必须是第一个参数。
五种通知执行的顺序
a. 在目标方法没有抛出异常的情况下:
前置通知
环绕通知(环绕通知调用目标方法之前代码、目标方法、环绕通知调用目标方法之后代码)
后置通知
最终通知
b.在目标方法抛出异常的情况下:
前置通知
环绕通知(环绕通知调用目标方法之前代码、目标方法抛出异常)
异常通知
最终通知
c.如果存在多个切面:
多切面执行时,采用了责任链设计模式。
切面的配置顺序决定了切面的执行顺序,多个切面执行的过程,类似于方法调用的过程,在环绕通知的proceed()执行时,去执行下一个切面或如果没有下一个切面执行目标方法。
五种通知常见使用场景
前置通知:记录日志(方法被调用)
环绕通知:控制事务、权限控制
后置通知:记录日志(方法已成功调用)
异常通知:异常处理、控制事务
最终通知:记录日志(方法已调用,但不一定成功)