业务场景:
外层一个大的事务,嵌套一个内层事务(如单据号获取后即递增更新到数据库),内层事务需独立提交(其原子性为了避免并发问题),不受外层事务影响(回滚或提交)。
现使用hibernateTransactionManager管理事务;
外层事务propagation="REQUIRED"
内层事务propagation="REQUIRES_NEW"
执行时内层事务可以顺利提交,但是外层事务老是抛乐观锁异常(项目中采用乐观锁机制),即内层的事务涉及的更新语句在外层事务提交时又去执行一次了。。。
网上查半天都没有看到类似的问题解决方案,有人遇到同样问题吗?该如何解决?
外层一个大的事务,嵌套一个内层事务(如单据号获取后即递增更新到数据库),内层事务需独立提交(其原子性为了避免并发问题),不受外层事务影响(回滚或提交)。
现使用hibernateTransactionManager管理事务;
外层事务propagation="REQUIRED"
内层事务propagation="REQUIRES_NEW"
执行时内层事务可以顺利提交,但是外层事务老是抛乐观锁异常(项目中采用乐观锁机制),即内层的事务涉及的更新语句在外层事务提交时又去执行一次了。。。
网上查半天都没有看到类似的问题解决方案,有人遇到同样问题吗?该如何解决?
A1:
按你的描述,你的事务配置是正确的:
外层事务propagation="REQUIRED"
内层事务propagation="REQUIRES_NEW"
你可以检查你配置的事务是否真由Spring托管,不是在代码中提交。
外层事务propagation="REQUIRED"
内层事务propagation="REQUIRES_NEW"
你可以检查你配置的事务是否真由Spring托管,不是在代码中提交。
A2:
非常感谢您的回复!
我们的事务都是使用Spring AOP的方式切入,不存在代码提交的情况。
如果内层事务也配置成propagation="REQUIRED",是正常的,只有在并发的情况下才偶尔会有乐观锁异常抛出,
内层事务尝试了AOP的方式切入,注解方式,HibernateTemplate().getSessionFactory().openSession()等方式,都同样只要一调用就有乐观锁异常。
另:我们用的是spring3.0, hibernate3.0。会不会hibernateTransactionManager就是不支持这样的事务嵌套。
我们的事务都是使用Spring AOP的方式切入,不存在代码提交的情况。
如果内层事务也配置成propagation="REQUIRED",是正常的,只有在并发的情况下才偶尔会有乐观锁异常抛出,
内层事务尝试了AOP的方式切入,注解方式,HibernateTemplate().getSessionFactory().openSession()等方式,都同样只要一调用就有乐观锁异常。
另:我们用的是spring3.0, hibernate3.0。会不会hibernateTransactionManager就是不支持这样的事务嵌套。
final Answer:
问题解决!
问题出在,内层事务更新之前有先把对象查询出来,而这个查询的方法没有被只读事务切到,
导致查询出来的对象改变后,外层事务提交时会重新执行更新.
问题出在,内层事务更新之前有先把对象查询出来,而这个查询的方法没有被只读事务切到,
导致查询出来的对象改变后,外层事务提交时会重新执行更新.
解决办法:业务逻辑层
1、查询方法添加只读事务@Transactional(propagation=Propagation.REQUIRES_NEW,readOnly=true)
2、更新方法添加@Transactional(propagation=Propagation.REQUIRES_NEW)
用例层的aop切入配置不变。