spring之事务的传播行为(三)
(1)当事务被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事物,并在自己的事务中运行。 (2)事务的传播行为可以由传播属性指定,spring定义了7种传播行为,最常用的是REQUIRED和REQUIRED_NEW。 在上一节我们利用事务解决了购买时候的问题,本节继续介绍事务的传播行为。 新建Cashier.java package com.gong.spring.tx; import java.util.List; public interface Cashier { void checkout(String username,List<String> isbns); } 新建CashierImpl.java org.springframework.beans.factory.annotation.Autowired; org.springframework.stereotype.Service; org.springframework.transaction.annotation.Transactional; @Service("cashier") class CashierImpl implements Cashier { @Autowired private BookShopService bookShopService; @Transactional @Override isbns) { for(String isbn: isbns){ bookShopService.purchase(username,isbn); } } } 意思是checkout是一个添加了事务的方法,而在该方法里调用了一个也添加了事务的方法bookShopService,那么在checkout中到底是用bookShopService中的事务,还是用自己本身的事务?我们以实际结果来看。 现在重新设置数据库中的数据: ? 我们在SpringTransactionTest中测试checkout方法: private Cashier cashier = null; cashier = ctx.getBean(Cashier.class); @Test void testTransactionlPropagation(){ cashier.checkout("AA",Arrays.asList("1001","1002")); } 两本都是可以购买成功的: ? 此时,我们再次运行SpringTransactionTest中的testTransactionlPropagation方法: 会抛出异常: ? 这是因为买了第一本之后剩30,不够买第二本,我们看看数据库中的结果: ? 却发现一本也没有买成功,这就是事务默认的传播行为,即在现有的事务内继续运行,也就是purcase方法上的注解实际默认是@Transactional(propagation=Propagation.REQUIRED)。因此checkout和bookShopService自始至终都在一个事务中,这个事务只在checkout结束的时候被提交,因此用户一本书都买不到。 使用@Transactional(propagation=Propagation.REQUIRES_NEW)来表示方法新开一个事务,如果该方法被另一个事务方法所调用,那么调用的事务方法就暂时被挂起。也就是说,为purchase方法加上了@Transactional(propagation=Propagation.REQUIRES_NEW)之后,purchase会在自己的事务中运行,并且在运行完之后,再运行checkout的事务。 我们为purchase上加上@Transactional(propagation=Propagation.REQUIRES_NEW)注解,再看下结果: ? 可以成功的买到一本。? (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |