面试官让列举Spring的事务会失效的场景,我说了8个

本文分享自华为云社区《哪些场景下Spring的事务会失效?》,作者:冰 河 。

在日常工作中,如果对Spring的事务管理功能使用不当,则会造成Spring事务不生效的问题。而针对Spring事务不生效的问题,也是在跳槽面试中被问的比较频繁的一个问题。

今天,我们就一起梳理下有哪些场景会导致Spring事务失效。

Spring事务不生效总览

简单来说,Spring事务会在几种特定的场景下失效,如下图所示。

cke_143.png

数据库不支持事务

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){

logger.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){

logger.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 异常类及其子类。

点击关注,第一时间了解华为云新鲜技术~

优秀的个人博客,低调大师

微信关注我们

原文链接:https://my.oschina.net/u/4526289/blog/10105171

转载内容版权归作者及来源网站所有!

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。

相关文章

发表评论

资源下载

更多资源
优质分享Android(本站安卓app)

优质分享Android(本站安卓app)

近一个月的开发和优化,本站点的第一个app全新上线。该app采用极致压缩,本体才4.36MB。系统里面做了大量数据访问、缓存优化。方便用户在手机上查看文章。后续会推出HarmonyOS的适配版本。

Mario,低调大师唯一一个Java游戏作品

Mario,低调大师唯一一个Java游戏作品

马里奥是站在游戏界顶峰的超人气多面角色。马里奥靠吃蘑菇成长,特征是大鼻子、头戴帽子、身穿背带裤,还留着胡子。与他的双胞胎兄弟路易基一起,长年担任任天堂的招牌角色。

Java Development Kit(Java开发工具)

Java Development Kit(Java开发工具)

JDK是 Java 语言的软件开发工具包,主要用于移动设备、嵌入式设备上的java应用程序。JDK是整个java开发的核心,它包含了JAVA的运行环境(JVM+Java系统类库)和JAVA工具。

Sublime Text 一个代码编辑器

Sublime Text 一个代码编辑器

Sublime Text具有漂亮的用户界面和强大的功能,例如代码缩略图,Python的插件,代码段等。还可自定义键绑定,菜单和工具栏。Sublime Text 的主要功能包括:拼写检查,书签,完整的 Python API , Goto 功能,即时项目切换,多选择,多窗口等等。Sublime Text 是一个跨平台的编辑器,同时支持Windows、Linux、Mac OS X等操作系统。