经常使用Spring AOP 做切面, 但是会遇到一个问题. 当在同一个类中, 调用当前类下其他方法时候, 会导致切面失效. 即:类自调用问题
问题描述:
例如: B()调用 A() 方法. 按照第一直觉, B()中调了A() 之后, 因为A()有事务注解, 事务是会生效的. 但实际不会生效.
public class OrderService{
@Transactional
public void methodA(){
}
public void methodB(){
methodA();
}
}
Spring 生成代理类. 大致如下.
public class OrderServiceProxyXxxx{
OrderService orderService;
public void methodA(){
beginTransaction();
orderService.methodA();
endTransaction();
}
public void methodB(){
orderService.methodB();
}
}
问题定位:
1. 先调用B()方法. 再B()中调用A():
由于沒有增加@Transactional. 代理类中实际是直接调用目标类中的方法. 目标类中B() 再调用目标类A().
调用顺序: OrderServiceProxyXxxx.methodB() > orderSerivce.methodB() > orderSerivce.methodA()
这个过程中可以看到. 实际沒有事务参与其中.
2. 反过来,先调用A(). 再A()中 调用B().
调用顺序:OrderServiceProxyXxxx.methodA() > orderSerivce.methodA() > orderSerivce.methodB()
这里由于OrderServiceProxyXxxx.MethodA() 织入了事务, 因此事务是有效的.
解决方法:
1. 不要再内部自调用方法中使用AOP. 如果是事务这种注解, 则放到最外层. 或者标记到类上.
这里还需要注意代理方式, 是JDK还是 cglib的方式. jdk 是基于接口的方式.对非public是无效的. 需要注意下.springboot2之后默认是cglib
2. 内部自调用, 使用代理对象调用.
1. 方法1: 使用@Autowired 注入自身对象. 有内部方法调用时, 通过注入的代理类调用. 而不是this 调用内部方法.
public class OrderService{
@Autowired OrderService orderService;
@Transactional
public void methodA(){
}
public void methodB(){
orderService.methodA();
}
}
2. 方法2:使用Spring 获取代理对象
public class OrderService{
@Transactional
public void methodA(){
}
public void methodB(){
((OrderService) AopContext.currentProxy()).methodA();
}
}
本文探讨了在Spring AOP中遇到的一个常见问题:当类内部方法互相调用时,@Transactional注解可能导致事务失效。通过分析问题定位,指出问题在于自调用未经过代理类,从而跳过了事务管理。文章提供了两种解决方案,包括将事务注解置于最外层或在类级别,以及利用Spring获取代理对象进行内部调用。
974

被折叠的 条评论
为什么被折叠?



