@Transactional注解的一个很容易被忽略的错误用法

@Transactional注解的一个很容易被忽略的错误用法

今日审查代码时发现对@Transactional注解的一个如下用法:

less 复制代码
//StockController.java

@AutoWired
private StockService stockService

//商品出库
@PostMapping("/product/stock/out-stock")
public Boolean productOutStock(@RequestBody StockChangeDTO data) {
  return stockService.outStock(data);
}

//StockService服务
public Boolean outStock(StockChangeDTO data) {
  //此处省略相关数据检查的前置代码
  ..........
  
  //执行出库的逻辑,acOut表示出库操作
  return this.doProductStockChange(data, acOut)
}

//执行出库的逻辑
@Transactional(rollback = Exception.class)
public Boolean doProductStockChange(StockChangeDTO data, StockChangeAction action) {
  //记录出入库记录
  
  //改变商品库存
  
  //省略其他逻辑代码
  .............
}

这段代码的逻辑是在StockService.outStock方法中调用了一个公用方法doProductStockChange完成出库逻辑,在doProductStockChange方法中涉及多笔数据操作,有记录出入库记录,有更改商品库存数据等等,为保证数据的一致性,这里添加了事物注解@Transactional来保证数据的一致性,乍一看好像没什么问题。

但是仔细一思考,这么用真的能保证事物能生效吗?

了解@Transactional注解的原理,当我们在方法上使用 @Transactional 注解时,Spring 将会创建一个代理对象来包装该方法。该代理对象将在该方法执行之前和之后添加一些代码,以启动和提交/回滚事务。如下图所示:

也就是说我们在StockController中注入StockService的Bean时,如果StockService中没有用到@Transactional之类的注解,那么注入的就是StockService服务本身的bean,相反如果有这类的注解时,那么注入的就不再是StockService本身的bean了,而是经过包装了一层的代理对象,这个代码对象在方法调用前会启动事物,在方法调用后提交事物,发生异常时执行事物回滚,这样就能保证事物是生效的。

我们再来看看StockService.java中outStock方法调用this.doProductStockChange方法,虽然doProductStockChange方法添加了事物注解,但是由于这个时候的this对象就是StockService本身而不是代理对象,因此事物就是无效的,这段代码并不能保证出入库时的数据一致性。

解决办法有以下几种:

1、在StockService.java中注入自己,然后调用doProductStockChange方法改为调用注入对象的;

kotlin 复制代码
//StockService.java

@AutoWired
private StockService stockService

public Boolean outStock(StockChangeDTO data) {
  //此处省略相关数据检查的前置代码
  ..........
  
  //执行出库的逻辑,acOut表示出库操作
  return stockService.doProductStockChange(data, acOut)
}

2、将outStock方法中的数据检查代码前置到controller中去检查,然后直接调用stockService.doProductStockChange方法。

第一种方法属于为了临时快速解决问题的偏门方法,对代码的可阅读性造成伤害,个人建议还是第二种方法更好更值得推荐使用。 仔细检查下代码中是否还有其他注解类似的错误,比如@ASync, @Cacheable、@CachePut、@CacheEvict, @Scheduled等等。

相关推荐
星辰徐哥6 小时前
Spring Boot 微服务架构设计与实现
spring boot·后端·微服务
星辰徐哥6 小时前
Spring Boot 数据导入导出与报表生成
spring boot·后端·ui
明夜之约6 小时前
Spring Boot 自动装配源码
java·spring boot·后端
Leaton Lee6 小时前
Spring Boot分层架构详解:从Controller到Service再到Mapper的完整流程
java·spring boot·后端·架构
Micro麦可乐6 小时前
Spring Boot 实战:从零设计一个短链系统(含完整代码与数据库设计)
数据库·spring boot·后端·哈希算法·雪花算法·短链系统
Jinkxs6 小时前
Resilience4j- 与 Spring Boot 快速集成:自动配置与基础注解使用
java·spring boot·后端
毕设源码_郑学姐6 小时前
计算机毕业设计springboot网络相册设计与实现 基于Spring Boot框架的在线相册管理系统开发与应用 Spring Boot驱动的网络影集设计与实践
spring boot·后端·课程设计
辣机小司6 小时前
【踩坑记录:Spring Boot 配置文件读取值不一致?警惕 YAML 的“八进制陷阱”与 SnakeYAML 版本之谜】
java·spring boot·后端·yaml·踩坑记录
一条小锦吕*6 小时前
基于Spring Boot + 数据可视化 + 协同过滤算法的推荐系统设计与实现(源码+论文+部署全讲解)
spring boot·算法·信息可视化
Jinkxs6 小时前
Prometheus - 监控微服务:Spring Boot 应用指标暴露与监控
spring boot·微服务·prometheus