1.当事务方法被另一个事务方法调用的时候,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。
2.事务的传播行为可以由传播属性指定,Spring定义了7种传播行为:
required: 如果有事务在运行,则当前方法就在这个事务内运行,否则就新启动一个事务,并在自己的事务内运行(默认)
required_new:当前方法必须启动新的事务,并在自己的事务内运行,如果有事务正在运行,应该将它挂起
supports:如果有事务在运行,当前方法在这个事务内运行,否则它可以不运行在事务中
not_supported:当前方法不应该运行在事务中,如果有运行的事务,将它挂起
mandatory:当千年方法必须运行在事务内部,如果么有正在运行的事务,就抛出异常
never:当前方法不应该运行在事务中,如果有运行事务,就抛出异常
nested: 如果有事务在运行,当前的方法就i应该在这个事务的嵌套事务内运行,否则,就启动一个新的事务,并在它自己的事务内运行。
package com.hcx.spring.tx;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService {
@Autowired
private BookShopDao bookShopDao;
//添加事务注解
//1指定事务的传播行为,即当前事务方法被另外一个事务方法调用时,如何使用事务。
//默认required,
//2使用isolation指定事务的隔离级别,常用的为读已提交
//3默认情况下Spring的生命式事务对所有的运行时异常进行回滚,也可以通过对应的属性
//进行设置
// @Transactional(propagation=Propagation.REQUIRES_NEW,
// isolation=Isolation.READ_COMMITTED,
// noRollbackFor= {UserAccountException.class},
// )
//4.使用readOnly,指定事务是否为只读(可以优化数据库引擎)
//5.timeout:单位秒,事务在强制回滚前可占用时间,
@Transactional(propagation=Propagation.REQUIRES_NEW,
isolation=Isolation.READ_COMMITTED,
readOnly=false,
timeout=3
)
@Override
public void purchase(String username, String isbn) {
//1.获取书单价
int price=bookShopDao.findBookPriceByIsbn(isbn);
//2.更新书库存
bookShopDao.updateBookStock(isbn);
//3.更新用户余额
bookShopDao.updateUserAccount(username, price);
}
}
在买书Service上再定义一个买书事务,可以买多本。这样就有两个事务嵌套。
假设买两本,第一本书能买,第二本库存不足。
BookService中的事务,默认使用propagation=Propagation.REQUIRED。在上层Cashier事务中执行所有过程,所以2本都买不到。
若使用propagation=Propagation.REQUIRES_NEW。则在Service中每次调用方法都会用新事务,则第一本能买,第二本不能。
@Service("cashier")
public class CashierImpl implements Cashier {
@Autowired
private BookShopService bookShopService;
@Transactional
@Override
public void checkout(String username, List<String> isbns) {
for(String isbn: isbns){
bookShopService.purchase(username, isbn);
}
}
}