spring之事务的传播行为(三)

(1)当事务被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事物,并在自己的事务中运行。

(2)事务的传播行为可以由传播属性指定,spring定义了7种传播行为,最常用的是REQUIRED和REQUIRED_NEW。

在上一节我们利用事务解决了购买时候的问题,本节继续介绍事务的传播行为。

新建Cashier.java

package com.gong.spring.tx;

import java.util.List;

public interface Cashier {

    public void checkout(String username, List isbns);
    
}

新建CashierImpl.java

package com.gong.spring.tx;

import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service("cashier")
public class CashierImpl implements Cashier {

    @Autowired
    private BookShopService bookShopService;
    
    @Transactional
    @Override
    public void checkout(String username, List 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
public void testTransactionlPropagation(){
    cashier.checkout("AA", Arrays.asList("1001", "1002"));
}

两本都是可以购买成功的:

 

此时,我们再次运行SpringTransactionTest中的testTransactionlPropagation方法:

会抛出异常:

spring之事务的传播行为(三)_第1张图片

 

这是因为买了第一本之后剩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)注解,再看下结果:

 

可以成功的买到一本。 

你可能感兴趣的:(spring之事务的传播行为(三))