1前置通知
在目标方法执行之前执行执行的通知
前置通知方法,可以没有参数,也可以额外接收一个JoinPoint,Spring会自动将该对象传入,代表当前的连接点,通过该对象可以获取目标对象 和 目标方法相关的信息。
注意,如果接收JoinPoint,必须保证其为方法的第一个参数,否则报错。(方法可接收参数JoinPoint jp )
Object obj=jp.getTarget();//获取目标对象
MethodSignature ms=(MethodSignature) jp.getSignature();//获取方法
Method method=ms.getMethod();//获取方法,(导反射包的)
ms.getReturnType();//返回类型
ms.getExceptionTypes();//异常类型
ms.getName();//方法名
ms.getParameterTypes();//参数类型
ms.getParameterNames();//参数名
<!-- 前置通知 -->
<aop:before method="before" pointcut-ref="pc01" />
2环绕通知
在目标方法执行之前和之后都可以执行额外代码的通知。
在环绕通知中必须显式的调用目标方法,否则目标方法不会执行。
这个显式调用时通过ProceedingJoinPoint来实现的,可以在环绕通知中接收一个此类型的形参,spring容器会自动将该对象传入,这个参数必须处在环绕通知的第一个形参位置。
**要注意,只有环绕通知可以接收ProceedingJoinPoint,而其他通知只能接收JoinPoint。
<!-- 环绕通知 -->
<aop:around method="around" pointcut-ref="pc02"/>
环绕通知需要返回返回值,否则真正调用者将拿不到返回值,只能得到一个null。
public Object around(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("环绕通知around");
Object obj = pjp.proceed();// 执行直接的被代理者的方法.不调用的话,则不会执行
return obj;
}
环绕通知有
控制目标方法是否执行、目标方法执行之前或之后执行额外代码、有控制是否返回值、甚至改变返回值
的能力
环绕通知虽然有这样的能力,但一定要慎用,要小心不要破坏了软件分层的“高内聚 低耦合”的目标。
3 后置通知
在目标方法执行之后执行的通知。
在后置通知中也可以选择性的接收一个JoinPoint来获取连接点的额外信息,但是这个参数必须处在参数列表的第一个。在后置通知中,还可以通过配置获取返回值。
<!-- 后置通知 -->
<aop:after-returning method="afterreturning" pointcut-ref="pc01" returning="msg"/>
public void afterreturning(JoinPoint jp, Object msg) {
System.out.println("后置通知");
}
4 异常通知
在目标方法抛出异常时执行的通知
可以配置传入JoinPoint获取目标对象和目标方法相关信息,但必须处在参数列表第一位
另外,还可以配置参数,让异常通知可以接收到目标方法抛出的异常对象
<!-- 异常通知 -->
<aop:after-throwing method="afterthrowing" pointcut-ref="pc01" throwing="e"/>
public void afterthrowing(JoinPoint jp, Throwable e) {
System.out.println("异常通知:" + e.getMessage());
}
5 最终通知
是在目标方法执行之后执行的通知。和后置通知不同之处在于,后置通知是在方法正常返回后执行的通知,如果方法没有正常返-例如抛出异常,则后置通知不会执行。而最终通知无论如何都会在目标方法调用过后执行,即使目标方法没有正常的执行完成。另外,后置通知可以通过配置得到返回值,而最终通知无法得到。
最终通知也可以额外接收一个JoinPoint参数,来获取目标对象和目标方法相关信息,但一定要保证必须是第一个参数。
<!-- 最终通知 -->
<aop:after method="after" pointcut-ref="pc01" />
public void after(JoinPoint jp) {
System.out.println("最终通知");
}
五种通知执行的顺序
在目标方法没有抛出异常的情况下
*前置通知*
*环绕通知的调用目标方法之前的代码*//取决于配置顺序
目标方法
*环绕通知的调用目标方法之后的代码*
*后置通知*//取决于配置顺序
最终通知//最终通知 后置通知和环绕通知结束取决于配置反向顺序
在目标方法抛出异常的情况下:
*前置通知*
*环绕通知的调用目标方法之前的代码*//取决于配置顺序
目标方法 抛出异常
异常通知
最终通知
如果存在多个切面:
多切面执行时,采用了责任链设计模式。
切面的配置顺序决定了切面的执行顺序,多个切面执行的过程,类似于方法调用的过程,在环绕通知的proceed()执行时,去执行下一个切面或如果没有下一个切面执行目标方法从而达成了如下的执行过程:

如果目标方法抛出异常:

五种通知的常见使用场景
前置通知 记录日志(方法将被调用)
环绕通知 控制事务 权限控制
后置通知 记录日志(方法已经成功调用)
异常通知 异常处理 控制事务
最终通知 记录日志(方法已经调用,但不一定成功)