用spring做事务控制有一段时间了,习惯了在方法上加@Transactional注解,今天发现一个问题,在这里记录一下,下面是一段伪代码来描述业务场景
public class OrderServiceImpl implements OrderService{ public BaseResponse confirmOrder(){ //do something //新增订单数据,需要做事务控制 addOrder (); //do other } @Transactional(rollbackFor = Exception.class, isolation = Isolation.READ_COMMITTED) public int addOrder(Order order){ ... } }
由于需要做事务控制的代码只有一段,故将此部分代码抽出来添加事务控制,但是由于调用addOrder方法默认是this.addOrder调用,并不是通过代理对象调用的,故spring是无法做事务控制的。
改进方式1
添加一个dao层,用于专门做事务控制,所有需要事务控制的方法在此层实现
改进方式2
将直接调用 addOrder() 改成 ((OrderService) AopContext.currentProxy()).addOrder() ,其中OrderService是当前类实现的接口。
这样就是通过代理对象来执行,不会影响事务。因为是内部的自我调用,所以需要在spring配置中添加
<aop:aspectj-autoproxy expose-proxy="true"/>
否则会抛出如下错误
java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available
实际开发中,service可能是对外暴露的协议,如果不想将此方法暴露出去,可以再实现一个新接口,在新接口中添加此方法,java虽然不允许多继承,但是可以实现多个接口。