Spring——AOP之通知
一、概述
1.在具体的连接点上要执行的操作。
2.一个切面可以包括一个或者多个通知。
3.通知所使用的注解的值往往是切入点表达式。
二、当前连接点细节
1.切入点表达式通常都会是从宏观上定位一组方法,和具体某个通知的注解结合起来就能够确定对应的连接点。那么就一个具体的连接点而言,我们可能会关心这个连接点的一些具体信息,例如:当前连接点所在方法的方法名、当前传入的参数值等等。这些信息都封装在JoinPoint接口的实例对象中。
- JoinPoint
三、通知类型
1.前置通知
(1)在方法执行之前执行的通知
(2)使用@Before注解
@Before(value = "execution(* com.atguigu.dao.Impl.CalculatorImpl.div(..))")
public void before(JoinPoint jp) {
System.out.println("我是前置通知");
}
2.最终通知
(1)后置通知是在连接点完成之后执行的,即连接点返回结果或者抛出异常的时候
(2)使用@After注解
//最终通知
@After(value="execution(* com.atguigu.dao.Impl.CalculatorImpl.*(..))")
public void after() {
System.out.println("我是最终通知");
}
3.后置(返回)通知
(1)无论连接点是正常返回还是抛出异常,最终通知都会执行。如果只想在连接点返回的时候记录日志,应使用返回通知代替后置通知。
(2)使用@AfterReturning注解,在返回通知中访问连接点的返回值
①在返回通知中,只要将returning属性添加到@AfterReturning注解中,就可以访问连接点的返回值。该属性的值即为用来传入返回值的参数名称
②必须在通知方法的签名中添加一个同名参数。在运行时Spring AOP会通过这个参数传递返回值
③原始的切点表达式需要出现在pointcut属性中
//后置通知(返回通知)
@AfterReturning(value="execution(* com.atguigu.dao.Impl.CalculatorImpl.*(..))",returning="result")
public void afterReturning(JoinPoint jp,Object result) {
String methodName = jp.getSignature().getName();
Object[] args = jp.getArgs();
System.out.println("方法名:"+methodName+",参数:"+Arrays.toString(args)+",返回结果是:"+result);
System.out.println("我是后置(返回)通知");
}
4.异常通知
(1)只在连接点抛出异常时才执行异常通知
(2)将throwing属性添加到@AfterThrowing注解中,也可以访问连接点抛出的异常。Throwable是所有错误和异常类的顶级父类,所以在异常通知方法可以捕获到任何错误和异常。
(3)如果只对某种特殊的异常类型感兴趣,可以将参数声明为其他异常的参数类型。然后通知就只在抛出这个类型及其子类的异常时才被执行
//异常通知
@AfterThrowing(value="execution(* com.atguigu.dao.Impl.CalculatorImpl.*(..))",throwing="e")
public void afterThrowing(RuntimeException e) {
System.out.println(e.getMessage());
System.out.println("我是异常通知");
}
5.环绕通知
(1)环绕通知是所有通知类型中功能最为强大的,能够全面地控制连接点,甚至可以控制是否执行连接点。
(2)对于环绕通知来说,连接点的参数类型必须是ProceedingJoinPoint。它是 JoinPoint的子接口,允许控制何时执行,是否执行连接点。
(3)在环绕通知中需要明确调用ProceedingJoinPoint的proceed()方法来执行被代理的方法。如果忘记这样做就会导致通知被执行了,但目标方法没有被执行。
(4)注意:环绕通知的方法需要返回目标方法执行之后的结果,即调用 joinPoint.proceed();的返回值,否则会出现空指针异常。
//环绕通知
@Around(value="execution(* com.atguigu.dao.Impl.CalculatorImpl.*(..))")
public Object around(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("环绕通知");
Object proceed = pjp.proceed();
System.out.println("返回结果是:"+proceed);
System.out.println("最终通知");
return proceed;
}
注意:以上的环绕通知可以简单理解为以下几种通知的组合
//环绕通知
@Around(value="execution(* com.atguigu.dao.Impl.CalculatorImpl.*(..))")
public Object around(ProceedingJoinPoint pjp){
//前置通知
System.out.println("环绕通知");
Object proceed=null;
try {
proceed = pjp.proceed();
//返回通知
System.out.println("返回结果是:"+proceed);
} catch (Throwable e) {
//异常通知
e.printStackTrace();
}
//最终通知
System.out.println("最终通知");
return proceed;
}
还没有评论,来说两句吧...