批量新增操作为什么要加@Transactional注解

java 复制代码
@Override
@Transactional
public void batchAdd(List<ServeUpsertReqDTO> serveUpsertReqDTOList) {
    for (ServeUpsertReqDTO serveUpsertReqDTO : serveUpsertReqDTOList) {
        //1.校验服务项是否为启用状态,不是启用状态不能新增
        ServeItem serveItem = serveItemMapper.selectById(serveUpsertReqDTO.getServeItemId());
        //如果服务项信息不存在或未启用
        if (ObjectUtil.isNull(serveItem) || serveItem.getActiveStatus() != FoundationStatusEnum.ENABLE.getStatus()) {
            throw new ForbiddenOperationException("该服务未启用无法添加到区域下使用");
        }

        //2.校验是否重复新增
        Long count = lambdaQuery()
                .eq(Serve::getRegionId, serveUpsertReqDTO.getRegionId())
                .eq(Serve::getServeItemId, serveUpsertReqDTO.getServeItemId())
                .count();
        if (count > 0) {
            throw new ForbiddenOperationException(serveItem.getName() + "服务已存在");
        }

        //3.新增服务
        Serve serve = BeanUtil.toBean(serveUpsertReqDTO, Serve.class);
        //默认为服务项的价格
        serve.setPrice(serveItem.getReferencePrice());
        Region region = regionMapper.selectById(serveUpsertReqDTO.getRegionId());
        serve.setCityCode(region.getCityCode());
        baseMapper.insert(serve);
    }
}

为什么这段 batchAdd 方法需要添加 @Transactional 注解,以及它的具体作用。

核心问题:数据一致性

batchAdd 方法的目的是批量新增 一系列服务。在这个过程中,它执行了多个独立的数据库写操作(baseMapper.insert(serve))。

想象一下,如果在批量处理的过程中发生了以下意外情况:

  • 批量列表中有10条数据。
  • 程序成功插入了前5条。
  • 在插入第6条时,数据库突然断电、或者网络中断、或者代码抛出了异常(比如后面的业务逻辑有问题)。

如果没有 @Transactional 注解,会发生什么?

  • 数据库中会残留前5条成功插入的数据。
  • 第6条及以后的数据没有被插入。
  • 这导致了数据不一致的状态。你的业务数据可能因此变得混乱,例如,部分数据成功导入,部分失败,而系统却没有任何记录或回滚机制。

@Transactional 的作用:保证事务的ACID特性

@Transactional 注解的核心作用就是将一个方法包装在一个数据库事务 中。事务遵循 ACID 原则,对于我们这个场景来说,最重要的是 Atomicity(原子性)Consistency(一致性)

  • 原子性 (Atomicity) : 事务中的所有操作被视为一个不可分割的整体。要么全部成功 ,要么全部失败

    • batchAdd 中,要么列表里的所有 Serve 都被成功插入数据库,要么在发生任何错误时,所有 已插入的 Serve 都会被撤销(回滚),数据库回到方法执行前的状态。
  • 一致性 (Consistency): 事务执行前后,数据库的完整性约束没有被破坏。

    • 通过保证原子性,@Transactional 确保了批量新增操作不会导致数据处于部分完成的不一致状态。

这个方法中 @Transactional 的具体行为

batchAdd 方法被调用时,Spring 框架会拦截这个调用,并:

  1. 开启一个新的数据库事务
  2. 执行方法体内部的逻辑:循环、校验、插入。
  3. 如果方法正常执行完毕 :Spring 会自动提交 (Commit) 这个事务,所有在循环中执行的 insert 操作都会被永久保存到数据库中。
  4. 如果方法执行过程中抛出了异常 (例如 throw new ForbiddenOperationException(...)):Spring 会自动回滚 (Rollback) 这个事务。数据库会撤销在本次事务中所有已执行的 insert 操作,仿佛这个方法从未执行过一样。

为什么这个方法特别需要 @Transactional

  1. 多步写操作 :它包含了一个循环,循环内部会执行多次 insert。这是最典型的需要事务保护的场景。
  2. 存在前置校验:方法在插入前做了"服务项是否启用"和"是否重复"的校验。如果没有事务,在高并发场景下,可能会出现两个线程同时通过校验,然后都执行插入,导致重复数据(虽然你的校验能防止大部分情况,但事务是最后的保障)。更重要的是,如果校验通过后,插入时失败,事务可以保证不会留下脏数据。
  3. 业务逻辑要求:从业务角度看,"批量新增"这个操作本身就应该是一个原子操作。用户期望的是要么全部添加成功,要么全部失败。

总结

@Transactional 注解为 batchAdd 方法提供了事务保障 。它确保了在批量插入一系列服务记录时,整个过程是原子的、一致的。如果其中任何一个环节出了问题,所有已完成的操作都会被撤销,从而避免了部分成功、部分失败所导致的数据不一致问题,让批量操作的结果更加可靠和可预测。

简单来说,它就是为了保证:要么全成,要么全败

相关推荐
q***9943 小时前
Redis的Spring配置
数据库·redis·spring
S***y3963 小时前
MySQL视频
数据库·mysql
周杰伦fans3 小时前
[特殊字符] 代理模式超详细讲解 ——.NET
数据库·c#·代理模式
TDengine (老段)4 小时前
TDengine 转换函数 TO_JSON 用户手册
android·大数据·数据库·json·时序数据库·tdengine·涛思数据
2301_800256114 小时前
第七章 空间存储与索引 知识点梳理3(空间填充曲线)
数据库·笔记·sql·postgresql
冰封剑心4 小时前
MiniCPM-V-2_6 (4-bit 量化)使用
java·前端·数据库
小满、5 小时前
MySQL :存储引擎原理、索引结构与执行计划
数据库·mysql·索引·mysql 存储引擎
x***13395 小时前
SQL Server 创建用户并授权
数据库·oracle
JIngJaneIL5 小时前
智慧物业|物业管理|基于SprinBoot+vue的智慧物业管理系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·论文·智慧物业管理系统