环绕通知
环绕通知是所有通知类型中功能最为强大的, 能够全面地控制连接点. 甚至可以控制是否执行连接点.
对于环绕通知来说, 连接点的参数类型必须是 ProceedingJoinPoint . 它是 JoinPoint 的子接口, 允许控制何时执行, 是否执行连接点.
在环绕通知中需要明确调用 ProceedingJoinPoint 的 proceed() 方法来执行被代理的方法. 如果忘记这样做就会导致通知被执行了, 但目标方法没有被执行.
注意: 环绕通知的方法需要返回目标方法执行之后的结果, 即调用 joinPoint.proceed(); 的返回值, 否则会出现空指针异常
package com.learn.spring.aspectJ;
import java.util.Arrays;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
/**
* 日志切面
*/
@Component //标识为组件
@Aspect //标识为切面
@Order(3) //指定切面的优先级. 值越小,优先级越高. 标注@Order的切面比不标注@Order切面的优先级高
public class LoggingAspect {
/**
* 环绕通知:类似于动态代理的整个过程.
*/
//@Around("execution(* com.learn.spring.aspectJ.*.*(..))")
public Object aroundMethod(ProceedingJoinPoint pjp){
String methodName = pjp.getSignature().getName();
Object [] args = pjp.getArgs();
try {
System.out.println("The method "+methodName+" begins with " +Arrays.asList(args));
//执行目标方法
Object result = pjp.proceed();
System.out.println("The method "+methodName+" ends with " +result );
return result ;
} catch (Throwable e) {
e.printStackTrace();
System.out.println("The method "+methodName+"occurs exception :" + e );
}finally{
System.out.println("The method " + methodName+"ends .");
}
return null ;
}
}