spring mvc AOP 在service里面,嵌套调用一个触发切面方法,然而并没有触发切面方法执行,但在controller里面直接调用这个方法,就可以触发切面方法。在springmvc里面,使用到AOP--面向切面编程,在切面做一些统一的事情。
遇到的情况是这样的:
1:在controller里面直接调用service里面的save方法,那么就会触发切面的方法。
2:在controller里面调用service的其他方法,其他方法,最终会间接的调用这个save方法,这个时候,就不会触发切面方法。
首先,上面的切面表达式是没有问题的,肯定能切到对应的方法的。
这个切面,就是去统一刷新一下缓存,所以,只需要在涉及到修改数据库的时候,才需要执行这个切面的方法。如果,没修改数据库的话,就不需要去刷新缓存。就这么一个需求。
解决方法:
原来的间接调用save方法,是如下样式的。
因为这个save()方法就在同一个类里面所以,就可以直接调用,不嫌麻烦的,还可以在前面加个this。
这么使用的话,是不会触发切面方法执行的。
修改如下:在service类里面,使用resource注解,把自己再给引入一下,然后使用他去调用目标方法,触发切面方法。
在自己的类里面,再把自己给引入进来,在需要触发切面方法的时候,就是这个selfService.目标方法(),来触发切面去执行切面方法。
原理:
看别人说这个使用ioc注入进来的这个service已经变成了一个代理啦,这个AOP的实现就是这个代理模式来实现的,你要是在类里面使用this来调用方法的话,就跟这个代理没关系啦,切面的触发,都是service的对应代理去触发的,所以,你在service里面直接调用触发切面的方法,是达不到这个效果的,所以,需要使用这个ioc注入的代理对象,就会触发切面的方法啦。
看这个图的执行结果,可验证如下几个结论。
1,spring使用的aop是基于cglib来实现的。
2,交给IOC注入到容器的bean,我们使用的那个service是代理对象,不是原始对象。
注意:
你这个不是在自己的类里面把自己又注入一次么。所以,你在使用这个selfService的时候,小心点,避免陷入自己调用自己,然后,陷入无限循环的尴尬境地。我想老铁不应该弄出这么简单的bug吧。