1. 前置增强(又称前置通知)
前置增强使用@Befor注解标识,增强方法优先于目标方法执行。
前置增强方法:
@Before("execution(int mul(int,int))")//执行方法之前执行下面的方法
public void before(JoinPoint jp) {
Object object = jp.getTarget();//得到目标对象
Object [] args = jp.getArgs();//得到参数
String name = jp.getSignature().getName();
//前置增强内容
System.out.println(object.getClass().getName()+":The "+name+" method begins.");
System.out.println(object.getClass().getName()+":Parameters of the "+name+" method: ["+args[0]+","+args[1]+"]");
}
运行Test类:控制台输出:
2. 后置增强(又称后置通知)
后置增强用@After标识,增强方法在目标方法执行后执行,无论目标方法运行期间是否出现异常。
后置增强方法:
@After("execution(int mul(int,int))")//执行哪个方法的时候(之前)执行下面的方法,这里是乘法
public void after(JoinPoint jp) {
Object object = jp.getTarget();//得到目标对象
Object [] args = jp.getArgs();//得到参数
String name = jp.getSignature().getName();
//后置增强内容
System.out.println(this.getClass().getName()+":The "+name+" method ends.");
}
运行Test类:控制台输出:
3.返回增强(又称返回通知)
返回增强以@AfterReturning注解为标识,在目标方法正常结束后执行,可以获取目标方法的执行结果。
返回增强方法:
returning为目标方法执行完后的结果,与方法中Object a中参数的a相同
@AfterReturning(value="execution(int mul(..))",returning="a")
public void afterReturning(JoinPoint jp,Object a) {
Object object = jp.getTarget();
String name= jp.getSignature().getName();
//返回增强内容
System.out.println(this.getClass().getName()+":Result of the "+name +" method:"+a);
}
运行Test类:控制台输出:
4.异常增强(又称异常通知)
异常增强以@AfterThrowing注解为标识,目标方法抛出异常之后执行,可以访问到异常对象,且可以指定在出现哪种异常时才执行增强代码
先在目标类中定义一个异常:
package com.jd.calculator;
import org.springframework.stereotype.Service;
@Service
public class CalculatorService implements ICalculatorService {
@Override
public int mul(int a, int b) {
int result = a*b;
if(result==0) {
throw new RuntimeException("异常!乘积不能为0!");
}
System.out.println("乘法运算结束,结果为"+result);
return result;
}
}
并修改Test类中mul方法的参数为(1,0)。
异常增强方法:
@AfterThrowing(value="execution(int mul(..))", throwing="e")
public void afterThrowing(JoinPoint jp,Exception e) {
Object object=jp.getTarget();
String name= jp.getSignature().getName();
System.out.println(this.getClass().getName()+":Result of the "+name +" method:"+e);
}
运行Test类:控制台输出:
5.环绕增强(又称环绕通知)
环绕增强由@Around修饰,可以实现上述@Before,@After,@AfterReturning和@AfterThrowing四种增强效果,可以实现动态代理全过程。
注意:
@Around修饰的方法中有ProceedingJoinPoint类型的参数,该变量可以决定是否执行目标方法;
@Before、@After、@AfterRunning和@AfterThrowing修饰的方法没有返回值;而@Around修饰的方法必须有返回值,返回值为目标方法的返回值;
环绕增强方法:
@Around("execution(int mul(int,int))")
public Object around(ProceedingJoinPoint jp) {
Object result = null;
Object obj = jp.getTarget();
Object[] args = jp.getArgs();
String name = jp.getSignature().getName().toString();
try {
try {
//前置增强
System.out.println(obj.getClass().getName()+":The "+name+" method begins.");
System.out.println(obj.getClass().getName()+":Parameters of the "+name+" method: ["+args[0]+","+args[1]+"]");
//执行目标类中的方法
result = jp.proceed();
} finally {
//后置增强
System.out.println(obj.getClass().getName()+":The "+name+" method ends.");
}
//返回增强
System.out.println(obj.getClass().getName()+": The result of the "+name+" method is "+result);
} catch (Throwable e) {
//异常增强
System.out.println(obj.getClass().getName()+":The "+name+" method has a exception:"+e.getMessage());
}
return result;
}
运行Test类:设置Test类中mul的参数分别为不出现异常时的mul(1,1)和出现异常时的(1,0)控制台输出
不出现异常时:
出现异常时:
三、前置增强、后置增强、返回增强、异常增强的执行顺序
源码:
try {
try {
doBefore();// @Before注解所修饰的方法
method.invoke();// 执行目标对象内的方法
} finally {
doAfter();// @After注解所修饰的方法
}
doAfterReturning();// @AfterReturning注解所修饰的方法
} catch (Exception e) {
doAfterThrowing();// @AfterThrowing注解所修饰的方法
}
分析源码可知,先执行前置增强,然后执行目标方法,之后立即执行后置增强。因为后置增强在finally代码块里,所以无论是否出现异常,后置增强一定执行。如果出现异常,不再执行返回增强,下一步直接执行异常增强。如果不出现异常,则执行完后置增强后,执行返回增强,不执行执行异常增强