参考:
https://www.cnblogs.com/ynyhl/p/12066530.html.
解决问题:
上面的博客中有下面一句话,感觉有点抽象:
只有在代理对象之间进行调用时,可以触发切面逻辑。
而在同一个class中,方法B调用方法A,调用的是原对象的方法,而不通过代理对象。所以Spring无法切到这次调用,也就无法通过注解保证事务性了。
我的理解如下:代理方法A调用了原方法A之后,原方法A内部调用了没有经过代理的方法B,导致方法B上的事务注解失效。
原生类
package com.pd;
import org.springframework.transaction.annotation.Transactional;
public class Test {
public void A() {
System.out.println("进入方法A");
B();
}
@Transactional
public void B() {
System.out.println("进入方法B");
}
}
代理类
package com.pd;
public class TestProxy {
private Test test= new Test();
/**
* 方法A的代理
*/
public void A() {
test.A();
}
/**
* 方法B的代理
*/
public void B() {
System.out.println("开启事务");
test.B();
System.out.println("关闭事务");
}
public static void main(String[] args) {
TestProxy proxyOfTest = new TestProxy();
proxyOfTest.A();
}
}
执行main方法,可以看见B的代理方法并没有执行,切面没有生效,结果如下:
进入方法A
进入方法B
上面的博客中的解决办法,我比较喜欢第四个,自己注入自己,真是漂亮的办法。但是感觉还是不够完美,要是完全无感知就好了。
然后又参考了一下这几篇博客:
https://blog.csdn.net/posse_wang/article/details/76690772.
https://www.itqiankun.com/article/spring-aop-aspectj.
AspectJ这种方式好像能完美解决问题,但是:
编译前织入,只能生成开发模块的类的织入代码;
编译后织入,配置比较繁琐,不知道开源版的ide支持得好不好;
类加载时织入,这个启动时要指定aspectjweaver-1.8.13.jar包,否则事务也是不生效。
并且AspectJ这种方式改变了class,不知道调试时会不会出问题。以后需要时再实战一下吧。