上一篇讲了,aop代理执行时,所有匹配该方法的切面都已经收集完毕,接下来就是 以责任链的模式 去遍历切面集合,来挨个增强原生方法。
data:image/s3,"s3://crabby-images/9c909/9c9094e582449aa269771f9c85c142f2cba50ad6" alt=""
5. 代理执行链的调用
如果没找到匹配该方法的切面集合,那么就返回调用原方法
data:image/s3,"s3://crabby-images/46418/4641886539b25e057afb1ad16cb7d200a58e2d18" alt=""
找到了,就匹配该方法的切面集合,就进行切面执行链的调用
会先创建一个代理执行上下文对象MethodInvocation,把所有的切面,代理对象,原生方法,参数,等等传进去
data:image/s3,"s3://crabby-images/ac131/ac13134b2ae865bae0352f57a7911a41ca505a0b" alt=""
然后调用MethodInvocation对象的proceed()方法,jdk动态代理实现用的是ReflectiveMethodInvocation类
data:image/s3,"s3://crabby-images/c9622/c9622845269d731f765e884950bc83aa5a8140ed" alt=""
测试代码
为了更直观的 描述切面执行链的调用过程,这里准备点测试代码,把五种aop方法代理类型的注解都加上
被代理的类和被代理的方法
@Component
public class MyBeanServiceImpl implements MyBeanService {
@Override
public String testMyBean() {
System.out.println("原生方法被执行...");
return "MyBeanServiceImpl.testMyBean方法的返回值";
}
}
aop配置类
aop配置类,五种aop代理类型的类型都加上
@Aspect
@Component
public class MyAspect {
@Pointcut("execution(* com.lb.springboot_simple_starter.bean.aop.service.impl.MyBeanServiceImpl.*(..))")
public void pc2() {}
@Before(" pc2()")
public void before() {
System.out.println("前置通知");
}
@After(" pc2()")
public void after() {
System.out.println("后置通知");
}
@Around("pc2()")
public void around(ProceedingJoinPoint point) {
System.out.println("环绕通知,原方法执行前");
try {
System.out.println("环绕通知中:" + point.proceed());
} catch (Throwable throwable) {
throwable.printStackTrace();
}
System.out.println("环绕通知,原方法执行后");
}
@AfterThrowing(value = " pc2()", throwing = "e")
public void AfterThrowing(Exception e) {
System.out.println("异常通知");
MethodInvocation methodInvocation = ExposeInvocationInterceptor.currentInvocation();
e.printStackTrace();
}
@AfterReturning(value = "pc2()", returning = "result")
public void doAfterReturning(Object result) {
System.out.println("【返回后通知】 返回值:" + result);
}
}
运行;
从spring容器中获取MyBeanService代理对象,并调用testMyBean()。
@Test
public void testGeneratorAdvisor(){
ApplicationContext applicationContext = new AnnotationConfigApplicationContext("com.lb.springboot_simple_starter.bean.aop");
MyBeanService bean = applicationContext.getBean(MyBeanService.class);
bean.testMyBean();
System.out.println(bean);
//MyService bean = applicationContext.getBean(MyService.class);
//bean.getUser();
}
data:image/s3,"s3://crabby-images/af352/af3523b32321d7e860f342ce19cf0aa710274b72" alt=""
ReflectiveMethodInvocation.proceed().
接下来,所有切面与当前执行方法匹配后,进入ReflectiveMethodInvocation.proceed()
data:image/s3,"s3://crabby-images/21929/219291b07f777bbeda3fcd90df15b1d3005a1dfe" alt=""
可以看到有有一个默认的切面,加我们自己配置的五个切面:按照Around.class, Before.class, After.class, AfterReturning.class, AfterThrowing.class 注解所对应的切面类 排序。
然后这里有个currentInterceptorIndex变量,默认为-1, 来控制 从数组取出切面的下标。++后从数组取出切面对象。++后为0。
data:image/s3,"s3://crabby-images/5a704/5a704732dfc005f0acea713babf767c91be2ac63" alt=""
1.执行ExposeInvocationInterceptor
currentInterceptorIndex 初始为0,取出数组0下标对应的切面类 ExposeInvocationInterceptor对象
data:image/s3,"s3://crabby-images/28e5f/28e5f30efa39c435e392fb21f99fb97579fb5784" alt=""
这里千万要注意,在调用 实现MethodInterceptor接口的所有切面实现类的invoke方法时会传入this,也就是当前MethodInvocation对象
ExposeInvocationInterceptor.invoke()
在 把MethodInvocation对象对象放入到ExposeInvocationInterceptor的 ThreadLocal<MethodInvocation> invocation后
data:image/s3,"s3://crabby-images/bd362/bd362ddf5ed983281fef822a3d1bb6fa03595e1f" alt=""
调用当前MethodInvocation对象.proceed()方法,进行递归,代码又回到了这个
data:image/s3,"s3://crabby-images/fc37f/fc37fb6e0543e59f14ab64de67e8dfb781ab8864" alt=""
2.执行AspectJAroundAdvice
上一轮取出ExposeInvocationInterceptor后, currentInterceptorIndex 已经被++成0了,这里++ = 1,取出下边为1的切面对象 : AspectJAroundAdvice
AspectJAroundAdvice用来处理@Around注解的
data:image/s3,"s3://crabby-images/d069b/d069b7c0bb528132c5d33d1cd5c1344d33223509" alt=""
AspectJAroundAdvice.invoke
data:image/s3,"s3://crabby-images/d6834/d6834b2666eef8be89a96b729e10505a8c7e3766" alt=""
调用AspectJAroundAdvice类的父类 AbstractAspectJAdvice的invokeAdviceMethod方法
这里注意下父类 AbstractAspectJAdvice的invokeAdviceMethod方法 是专门用来执行 增强逻辑的方法,也就是有@Around,@Before,@After,@AfterReturning,@AfterThrowing 的方法。切面对象 是会存有 对应方法的Method对象
data:image/s3,"s3://crabby-images/2bd89/2bd897de354acc8f643522ea5b92be312d83a995" alt=""
然后反射调用 有@Around注解的方法
data:image/s3,"s3://crabby-images/08729/0872997d84a3b3f132dbc4209b09d45654868838" alt=""
这样就会被调用到我们自己写的@Around方法。
data:image/s3,"s3://crabby-images/1c587/1c5871b769ecd0f480aee38b7ccb5b54b9cd393a" alt=""
@Around注解下,我们一般会把增强的逻辑代码写在point.proceed()前后
point.proceed()方法 就又会递归到MethodInvocation的procceed()方法,把切面执行链传递下去。
千万注意,此时这个方法 还未出栈,也就是 point.proceed()方法后的方法 还要等 point.proceed()方法执行完后再执行。
data:image/s3,"s3://crabby-images/266ff/266ff45b329ef6fac38258b4ffdec28b5dedd337" alt=""
接下来代码又会回到MethodInvocation的procceed()。
data:image/s3,"s3://crabby-images/e06c7/e06c7b06a92c019eeff5a2bfc76976a4a10fb2bc" alt=""
3.执行MethodBeforeAdviceInterceptor
上一轮取出AspectJAroundAdvice后, currentInterceptorIndex 已经被++成1了,这里++ = 2,取出下边为2的切面对象 : MethodBeforeAdviceInterceptor
MethodBeforeAdviceInterceptor用来处理@Before注解的
data:image/s3,"s3://crabby-images/b4d4a/b4d4affd1281a84ec204cca701dac38ff276de57" alt=""
MethodBeforeAdviceInterceptor.invoke
这里处理@Before注解的AspectJMethodBeforeAdvice类 是经过适配的,由MethodBeforeAdviceInterceptor持有它,
MethodBeforeAdviceInterceptor.invoke会 先调用AspectJMethodBeforeAdvice的before(),去调用@Before的方法
data:image/s3,"s3://crabby-images/0705d/0705d41b93471797e159fe681cea6e04a0c078a2" alt=""
data:image/s3,"s3://crabby-images/0de12/0de120150229031694515b1f4fbf313c7d2796df" alt=""
这里由会调用到公用的 父类AbstractAspectJAdvice去执行advise对象里面 的增强方法,返回调用@Before方法
data:image/s3,"s3://crabby-images/0360d/0360d6216aa6ff133466ff1ce84d95177676fab8" alt=""
MyAspect.before()调用
data:image/s3,"s3://crabby-images/436d2/436d2810b8c473e0f649a359772c1bde3a509455" alt=""
接下来该方法结束,
代码回到 MethodBeforeAdviceInterceptor.invoke方法,又调用MethodInvocation得proceed(),把执行链传递下去
data:image/s3,"s3://crabby-images/11d0e/11d0e8a6ca631103d37a52b1ce1a63c70df2056c" alt=""
代码回到这里MethodInvocation.proceed()
data:image/s3,"s3://crabby-images/da424/da42421261e21d3f8ef70834c6be0ab1ded1fa4f" alt=""
4.执行 AspectJAfterAdvice
上一轮取出AspectJAroundAdvice后, currentInterceptorIndex 已经被++成2了,这里++ = 3,取出下边为3的切面对象 : AspectJAfterAdvice
AspectJAfterAdvice用来处理@After注解的
data:image/s3,"s3://crabby-images/dfed4/dfed4638a459afaef02edaf35892f7ddca7748e1" alt=""
AspectJAfterAdvice.invoke(mi)
@After切面的处理逻辑就有点奇特了,他先调用MethodInvocation的proceed(),把执行链传递下去,
再在finally代码块去 调用@After方法,也就是会等剩下的 @AfterReturning,@AfterThrowing切面执行完 以及原生方法 执行完,再去调用 调用@After方法
data:image/s3,"s3://crabby-images/3c633/3c633d4a062c96c2435223057e839f5b34799620" alt=""
这里AspectJAfterAdvice.invoke还没出栈,先把执行链传递下去,代码又回到MethodInvocation.proceed()
data:image/s3,"s3://crabby-images/b90ae/b90ae27c286ff3e65a0c7fa372dccd88b05fb5c2" alt=""
5.执行 AfterReturningAdviceInterceptor
上一轮取出AspectJAfterAdvice后, currentInterceptorIndex 已经被++成3了,这里++ = 4,取出下标为4的切面对象 : AfterReturningAdviceInterceptor
AfterReturningAdviceInterceptor用来处理@AfterReturning注解的
data:image/s3,"s3://crabby-images/c4e11/c4e11419b2853ee7bbc797c2756ce8c8ac5dc0c3" alt=""
AfterReturningAdviceInterceptor.invoke(mi)
这里处理@AfterReturning注解的AfterReturningAdvice类 是经过适配的,由AfterReturningAdviceInterceptor.invoke持有它,
AfterReturningAdviceInterceptor.invoke可以调用AfterReturningAdvice的afterReturning(),去调用@AfterReturning的方法。
@AfterReturning注解的作用就是 等原生方法返回之后,再执行@AfterReturning方法,所以这里原生方法还没调用,更不会有返回值,所以把执行链传递下去,等剩下的@AfterThrowing注解处理完,原生方法调用完获取到返回值之后,再执行 AfterReturningAdvice的afterReturning(),去调用@AfterReturning的方法。
data:image/s3,"s3://crabby-images/9bb8b/9bb8b092e9f13e1ba2aca01858862c17ed4e516e" alt=""
这里AfterReturningAdviceInterceptor.invoke(mi)还没出栈,先把执行链传递下去,代码又回到MethodInvocation.proceed()
data:image/s3,"s3://crabby-images/102f4/102f4019d69b6f9f359a302496033c2e1c2a9aac" alt=""
6.执行AspectJAfterThrowingAdvice
上一轮取出AfterReturningAdviceInterceptor后, currentInterceptorIndex 已经被++成4了,这里++ = 5,取出下标为5的切面对象 : AfterReturningAdviceInterceptor
AfterReturningAdviceInterceptor用来处理@AfterThrowing注解的
data:image/s3,"s3://crabby-images/cd284/cd284efe4ed042c8aae6ed0fa1cd8153736c5483" alt=""
AspectJAfterThrowingAdvice.invoke(md)
@AfterThrowing注解的作用就是 原生方法执行时抛出异常后,再执行@AfterThrowing方法,并传入异常对象并进行处理增强,所以这里原生方法还没调用,
所以先try_catch, 先调用MethodInvocation的proceed方法,把执行链传递下去,等原生方法调用抛出异常时,catch到异常,再执行 invokeAdviceMethod方法 ,去调用@AfterThrowing的方法。
data:image/s3,"s3://crabby-images/d4abc/d4abc719d733a51ed556562b150d3a2a93b317b2" alt=""
所以这里,代码又回到了MethodInvocation的proceed方法,注意此时AspectJAfterThrowingAdvice.invoke方法 并未出栈
data:image/s3,"s3://crabby-images/a1098/a109826156794844733c305cff81c095e20644de" alt=""
7.调用原生方法
我们一开始就只有6个切面,当上一个切面执行AspectJAfterThrowingAdvice 被取出时,currentInterceptorIndex被++,已经从4变到5了,所以再次进入MethodInvocation的proceed,会判断所以切面是否被执行完
判断 currentInterceptorIndex 是否等于 切面数组长度 -1,等于的话,说明所有切面都已经被调用了。
data:image/s3,"s3://crabby-images/920ce/920cef3e15389ab13dfff4019439a8aa59578ea0" alt=""
所有切面被调用了,那么接下来就只有调用原生方法了,也就是被代理方法
data:image/s3,"s3://crabby-images/25625/25625d5e5418b0c23b527ab2293e2f1c53858e13" alt=""
invokeJoinpoint()反射调用原生方法
就是反射调用 原生方法
data:image/s3,"s3://crabby-images/fc7dd/fc7ddfdc50c2727d6fceac900c2362d9c3453931" alt=""
传入被代理方法,反射调用method
data:image/s3,"s3://crabby-images/d32a0/d32a0f0cabd8fd35bdab1fe31d7df34349e5f0dd" alt=""
进入我们的原生方法,返回返回值,不抛出异常
data:image/s3,"s3://crabby-images/fafc6/fafc660e701b570c9e93d9953242bd3681860545" alt=""
然后方法结束
8.AspectJAfterThrowingAdvice.invoke(md)出栈
之前最后一次进入MethodInvocation.proceed()是在 AspectJAfterThrowingAdvice.invoke(md) 时,方法内部 调用的,所以,这里原生方法被执行return 后,回到 AspectJAfterThrowingAdvice.invoke(md)
这里没抛出异常,所以不会走catch里面的逻辑 : 判断抛出异常是否等于 @AfterThrowing里配置的异常,是的话,就调用@AfterThrowing方法,传入异常对象
data:image/s3,"s3://crabby-images/0ef71/0ef71c0815420ca35c599e9ebd3b4bcc1b3dc3f7" alt=""
这里return出去后,ReflectiveMethodInvocation.proceed 也return
data:image/s3,"s3://crabby-images/42b45/42b458864a69d280646ac0a16f9e7811c32897cd" alt=""
9.AfterReturningAdviceInterceptor.invoke(md)出栈
调用AspectJAfterThrowingAdvice.invoke的MethodInvocation.proceed ()是由AfterReturningAdviceInterceptor.invoke(mi)调用的,所以MethodInvocation.proceed () 返回后,代码回到 AspectJAfterThrowingAdvice.invoke(md)
data:image/s3,"s3://crabby-images/ed409/ed4094cce55a22574516343afa6b175bdeb4d387" alt=""
接下来调用AfterReturningAdvice的afterReturning方法,去调用@AfterReturning方法
data:image/s3,"s3://crabby-images/bfe6b/bfe6ba12f3f96b6a112f8685f1975a5e3273aaca" alt=""
invokeAdviceMethod肯定又是公用父类AbstractAspectJAdvice里面的,反射调用AfterReturningAdvice对象里存储的@@AfterReturning方法 的Method对象,并传入返回值
data:image/s3,"s3://crabby-images/c5de8/c5de8203ab06ca1eff87fab6ce91535123c33c5d" alt=""
data:image/s3,"s3://crabby-images/6d951/6d951c1358fd44f975b9124df558cc3b4f088b42" alt=""
然后@AfterReturning方法返回
AfterReturningAdviceInterceptor.invoke返回
data:image/s3,"s3://crabby-images/da6b2/da6b2f653a0c00e8b39c624551d8a0c7dcf5f645" alt=""
ReflectiveMethodInvocation.invoke方法返回
data:image/s3,"s3://crabby-images/3b888/3b8889d41bd7f47229408e33ee7741209a4baaf6" alt=""
10. AspectJAfterAdvice.invoke出栈
代码回到上一个调用 MethodInvocation.invoke()方法的AspectJAfterAdvice.invoke()方法内
data:image/s3,"s3://crabby-images/2e167/2e1678e154b55ddba89723442ae05ed97110bb75" alt=""
然后在finally块调用invokeAdviceMethod() ,invokeAdviceMethod()肯定又是公用父类AbstractAspectJAdvice里面的,去调用@After方法的Method对象,这里就不赘述了,直接进入到@After方法
这里和@AfterReturning方法的区别就是不在乎返回值
data:image/s3,"s3://crabby-images/9d9ef/9d9ef0b8675b70ec0c0947c822b4ea840a5a7dfb" alt=""
然后AspectJAfterAdvice.invoke出栈,调用它的 MethodInvocation.invoke()也随之出栈
data:image/s3,"s3://crabby-images/ee44b/ee44b2606e6e9a5a53bf242644a397bd8ebe6317" alt=""
data:image/s3,"s3://crabby-images/230c7/230c75427982e4070e92d5afa3ebad40d9b79e88" alt=""
12.MethodBeforeAdviceInterceptor.invoke(md)出栈
代码回到上一个 调用 MethodInvocation.proceed()的 MethodBeforeAdviceInterceptor.invoke(md)里
data:image/s3,"s3://crabby-images/cbcce/cbcced8da9222d89a6f069ea72b8c30c70aac112" alt=""
这里MethodInvocation.proceed(),MethodBeforeAdviceInterceptor.invoke(md)也return,又回到 MethodInvocation.proceed() 去return
data:image/s3,"s3://crabby-images/5a6df/5a6df0215e4813d060edef9b417d19ecac678b1e" alt=""
13.@Around方法出栈
上一次调用MethodInvocation.proceed() 是在我们@Around方法的内部,手动调用的,所以MethodInvocation.proceed() return后 代码又回到这,
data:image/s3,"s3://crabby-images/03633/036336c17945216a41b3c953b8e91f9d39ef5bb2" alt=""
接着执行point.proceed() 后面的代码,结束@Around方法
然后代码回到 调用@Around方法的AspectJAroundAdvice.invoke方法
data:image/s3,"s3://crabby-images/f9cad/f9cad213c99147694c0cbdf57f2f487569968a42" alt=""
MethodInvocation.proceed() return
data:image/s3,"s3://crabby-images/85a3f/85a3f9b2f011db313308645a8a7dc9453d23205f" alt=""
14.ExposeInvocationInterceptor出栈
代码回到最初 调用MethodInvocation.proceed()的 ExposeInvocationInterceptor.invoke中
直接return 执行完 MethodInvocation.proceed() 后执行finally, 把旧的 aop代理执行上下文 设置回去 ,方法结束,ExposeInvocationInterceptor.inoke方法也随之出栈结束了。
data:image/s3,"s3://crabby-images/2e504/2e504fbd88a95cafdfc1007ba20a1b8e8b303816" alt=""
15.JdkDynamicAopProxy.invoke 出栈
至此,所有的aop切面执行完了, JdkDynamicAopProxy 这个InvokeHandler的 invoke方法执行完,返回返回值,代理对象的代理逻辑也就彻底走完了。
data:image/s3,"s3://crabby-images/27002/27002e402468dcb644128d365dcfdc7b295131ad" alt=""
代理对象的方法 调用就彻底完成了
data:image/s3,"s3://crabby-images/57889/57889041142c9da39e295a401181c9e156d42cec" alt=""
网友评论