Spring Boot事务管理详解(附银行转账案例)

一、事务基础概念

事务的ACID特性

  • 原子性(Atomicity):操作要么全部成功,要么全部失败
  • 一致性(Consistency):数据在事务前后保持合法状态
  • 隔离性(Isolation):多个事务并发互不干扰
  • 持久性(Durability):事务提交后数据永久保存

二、Spring Boot事务实战

1. 环境准备

java 复制代码
// Maven依赖
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>

2. 实体类定义

java 复制代码
@Entity
public class Account {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String accountNumber;
    private BigDecimal balance;
    // 省略getter/setter
}

@Entity
public class TransferLog {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String fromAccount;
    private String toAccount;
    private BigDecimal amount;
    // 省略getter/setter
}

3. Service层事务控制

java 复制代码
@Service
public class BankService {

    @Autowired
    private AccountRepository accountRepository;
    
    @Autowired
    private TransferLogRepository transferLogRepository;

    // 核心事务方法
    @Transactional(rollbackFor = Exception.class)
    public void transferMoney(String fromAccNum, String toAccNum, BigDecimal amount) {
        // 1. 扣减转出账户
        Account fromAccount = accountRepository.findByAccountNumber(fromAccNum);
        fromAccount.setBalance(fromAccount.getBalance().subtract(amount));
        accountRepository.save(fromAccount);

        // 模拟异常(测试事务回滚)
        if(amount.compareTo(BigDecimal.ZERO) < 0) {
            throw new RuntimeException("转账金额不能为负数");
        }

        // 2. 增加转入账户
        Account toAccount = accountRepository.findByAccountNumber(toAccNum);
        toAccount.setBalance(toAccount.getBalance().add(amount));
        accountRepository.save(toAccount);

        // 3. 记录转账日志
        TransferLog log = new TransferLog();
        log.setFromAccount(fromAccNum);
        log.setToAccount(toAccNum);
        log.setAmount(amount);
        transferLogRepository.save(log);
    }
}

三、关键注解说明

@Transactional 参数解析:

java 复制代码
@Transactional(
    isolation = Isolation.DEFAULT,    // 事务隔离级别
    propagation = Propagation.REQUIRED, // 传播行为
    rollbackFor = Exception.class,    // 指定回滚的异常类型
    timeout = 30                     // 事务超时时间(秒)
)

四、常见事务失效场景

  1. 非public方法@Transactional只能用于public方法
  2. 自调用问题:同一个类中方法A调用方法B(B有事务注解),事务不生效
  3. 异常被捕获:事务方法内捕获异常未重新抛出
  4. 错误异常类型 :默认只回滚RuntimeException,需通过rollbackFor指定

五、事务传播机制示例

java 复制代码
// 嵌套事务示例
@Transactional(propagation = Propagation.REQUIRED)
public void parentMethod() {
    // 主事务逻辑
    
    childMethod();  // 嵌套子事务
}

@Transactional(propagation = Propagation.REQUIRES_NEW)
public void childMethod() {
    // 独立事务执行
    // 即使外层事务回滚,此方法仍会提交
}

六、事务验证测试

java 复制代码
@SpringBootTest
class TransactionTest {

    @Autowired
    private BankService bankService;

    @Test
    void testTransferSuccess() {
        // 正常转账测试
        bankService.transferMoney("A123", "B456", new BigDecimal("100.00"));
        // 验证账户余额和日志记录
    }

    @Test
    void testTransferRollback() {
        // 测试异常回滚
        assertThrows(RuntimeException.class, () -> {
            bankService.transferMoney("A123", "B456", new BigDecimal("-100.00"));
        });
        // 验证数据未修改
    }
}

最佳实践建议

  1. 事务方法尽量放在Service层
  2. 明确指定rollbackFor属性
  3. 避免长事务(复杂操作拆分为多个小事务)
  4. 结合@Transactional与数据库约束保证数据一致性

扩展学习

  • Spring官方文档:Transactions
  • 分布式事务解决方案:Seata、XA协议
  • 事务隔离级别深度解析(脏读/幻读/不可重复读)
相关推荐
Zephyrtoria1 小时前
区间合并:区间合并问题
java·开发语言·数据结构·算法
yuren_xia6 小时前
RabbitMQ 知识详解(Java版)
java·rabbitmq·java-rabbitmq
kfyty7256 小时前
轻量级 ioc 框架 loveqq,支持接口上传 jar 格式的 starter 启动器并支持热加载其中的 bean
java·jvm·ioc·jar·热加载
早起鸟儿7 小时前
docker-Dockerfile 配置
java·linux·运维·docker
云边小网安7 小时前
java集合篇(六) ---- ListIterator 接口
java·开发语言·青少年编程·java集合
都叫我大帅哥7 小时前
Spring WebFlux:响应式编程的“未来战士”还是“花架子”?
java·spring·flux
都叫我大帅哥7 小时前
Reactor 深度解析:响应式编程的「核反应堆」是如何工作的?
java·spring
不太厉害的程序员7 小时前
NC65配置xml找不到Bean
xml·java·后端·eclipse
Mr.stupidCoder7 小时前
Git将本地文件推送到GitHub仓库
git·elasticsearch·github
chanalbert8 小时前
数据库连接池深度研究分析报告
数据库·spring