事务 失效场景
简单来说,Spring事务会在几种特定的场景下失效,如下图:
数据库不支持事务
Spring事务生效的前提是所连接的数据库要支持事务,如果底层的数据库都不支持事务,则Spring的事务肯定会失效。
如,如果使用的数据库为MySQL
,且选用了MyISAM
存储引擎,则Spring的事务就会失效。
事务方法未被Spring管理
如果事务方法所在的类没有加载到Spring IOC容器中,也就是说,事务方法所在的类没有被Spring管理,则Spring事务会失效,示例如下。
public class ProductService {
@Autowired private ProductDao productDao;
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void updateProductStockCountById(Integer stockCount, Long id){
productDao.updateProductStockCountById(stockCount, id);
}
}
ProductService
类上没有标注@Service
注解,Product
的实例没有加载到Spring IOC
容器中,就会造成updateProductStockCountById()
方法的事务在Spring
中失效。
方法没有被public修饰
如果事务所在的方法没有被public修饰,此时Spring的事务会失效,例如,如下代码所示。
@Service
public class ProductService {
@Autowired private ProductDao productDao;
@Transactional(propagation = Propagation.REQUIRES_NEW)
private void updateProductStockCountById(Integer stockCount, Long id){
productDao.updateProductStockCountById(stockCount, id);
}
}
虽然ProductService
上标注了@Service
注解,同时updateProductStockCountById()
方法上标注了@Transactional(propagation = Propagation.REQUIRES\_NEW)
注解。
但是,由于updateProductStockCountById()
方法为内部的私有方法(使用private修饰),那么此时updateProductStockCountById()
方法的事务在Spring中会失效。
同一类中方法调用
如果同一个类中的两个方法分别为A和B,方法A上没有添加事务注解,方法B上添加了@Transactional
事务注解,方法A调用方法B,则方法B的事务会失效。例如,如下代码所示。
@Service
public class OrderService {
@Autowired
private OrderDao orderDao;
@Autowired
private ProductDao productDao;
public void submitOrder(){
//生成订单
Order order = new Order();
long number = Math.abs(new Random().nextInt(500));
order.setId(number);
order.setOrderNo("order_" + number);
orderDao.saveOrder(order);
//减库存
this.updateProductStockCountById(1, 1L);
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void updateProductStockCountById(Integer stockCount, Long id){
productDao.updateProductStockCountById(stockCount, id);
}
}
submitOrder()
方法和updateProductStockCountById()
方法都在OrderService
类中,submitOrder()
方法上没有标注事务注解,updateProductStockCountById()
方法上标注了事务注解,submitOrder()
方法调用了updateProductStockCountById()
方法,此时,updateProductStockCountById()
方法的事务在Spring中会失效。
未配置事务管理器
如果在项目中没有配置Spring
的事务管理器,即使使用了Spring
的事务管理功能,Spring
的事务也不会生效。
例如,没有在项目的配置类中配置如下代码。
@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
}
此时,Spring
的事务就会失效。
方法的事务传播类型不支持事务
如果内部方法的事务传播类型为不支持事务的传播类型,则内部方法的事务在Spring中会失效。
例如,如下代码所示。
@Service
public class OrderService {
@Autowired
private OrderDao orderDao;
@Autowired
private ProductDao productDao;
@Transactional(propagation = Propagation.REQUIRED)
public void submitOrder(){
//生成订单
Order order = new Order();
long number = Math.abs(new Random().nextInt(500));
order.setId(number);
order.setOrderNo("order_" + number);
orderDao.saveOrder(order);
//减库存
this.updateProductStockCountById(1, 1L);
}
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void updateProductStockCountById(Integer stockCount, Long id){
productDao.updateProductStockCountById(stockCount, id);
}
}
由于updateProductStockCountById()
方法的事务传播类型为NOT\_SUPPORTED
,不支持事务,则updateProductStockCountById()
方法的事务会在Spring
中失效。
不正确的捕获异常
不正确的捕获异常也会导致Spring的事务失效,示例如下。
@Service
public class OrderService {
@Autowired
private OrderDao orderDao;
@Autowired
private ProductDao productDao;
@Transactional(propagation = Propagation.REQUIRED)
public void submitOrder(){
//生成订单
Order order = new Order();
long number = Math.abs(new Random().nextInt(500));
order.setId(number);
order.setOrderNo("order_" + number);
orderDao.saveOrder(order);
//减库存
this.updateProductStockCountById(1, 1L);
}
@Transactional(propagation = Propagation.REQUIRED)
public void updateProductStockCountById(Integer stockCount, Long id){
try{
productDao.updateProductStockCountById(stockCount, id);
int i = 1 / 0;
}catch(Exception e){
log.error("扣减库存异常:", e.getMesaage());
}
}
}
updateProductStockCountById()
方法中使用try-catch
代码块捕获了异常,即使updateProductStockCountById()
方法内部会抛出异常,但也会被catch
代码块捕获到,此时updateProductStockCountById()
方法的事务会提交而不会回滚,并且submitOrder()
方法的事务会提交而不会回滚,这就造成了Spring事务的回滚失效问题。
错误的标注异常类型
如果在@Transactional注解中标注了错误的异常类型,则Spring事务的回滚会失效,示例如下。
@Transactional(propagation = Propagation.REQUIRED)
public void updateProductStockCountById(Integer stockCount, Long id){
try{
productDao.updateProductStockCountById(stockCount, id);
}catch(Exception e){
log.error("扣减库存异常:", e.getMesaage());
throw new Exception("扣减库存异常");
}
}
在updateProductStockCountById()
方法中捕获了异常,并且在异常中抛出了Exception
类型的异常,此时,updateProductStockCountById()
方法事务的回滚会失效。
为何会失效呢?这是因为Spring中对于默认回滚的事务异常类型为RuntimeException,上述代码抛出的是Exception
异常。
默认情况下,Spring事务中无法捕获到Exception异常,所以此时updateProductStockCountById()
方法事务的回滚会失效。
此时可以手动指定updateProductStockCountById()
方法标注的事务异常类型,如下所示。
@Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)
这里,需要注意的是:Spring事务注解@Transactional中的rollbackFor属性可以指定 Throwable 异常类及其子类。
- 感谢你赐予我前进的力量