【数据库基础】转账100块怎么丢了?通俗讲解数据库事务ACID特性

本文涉及的产品
RDS Agent(兼容OpenClaw),2核4GB
RDS AI 助手,专业版
RDS MySQL DuckDB 分析主实例,集群系列 4核8GB
简介: 本文深入浅出地讲解数据库事务的ACID四大特性。以转账场景为例,介绍事务“要么全成功,要么全失败”的核心思想。详解原子性(Undo Log回滚)、一致性(数据守恒)、隔离性(并发控制)与持久性(Redo Log保障),助你理解数据库可靠性的基石。

前言

在上一篇关于 InnoDB 的文章中,我们提到 InnoDB 相比 MyISAM 最大的优势之一就是支持事务(Transaction)

初学者往往觉得“事务”这个词很高大上,但其实它解决的问题非常接地气。

试想一个场景:A 向 B 转账 100 元。 在数据库层面,这其实是两步操作:

  1. A 的账户余额减去 100 元。
  2. B 的账户余额加上 100 元。

如果在第1步执行完后,服务器突然断电了,或者程序报错了,第2步没执行。结果就是:A 的钱少了,B 的钱也没收到,这就出大事了。

为了解决这个问题,数据库引入了事务的概念。简单来说,事务就是**“一组操作,要么全部成功,要么全部失败,决不允许只做一半”**。

为了衡量一个数据库是否支持事务,计算机科学家提出了著名的 ACID 四大特性。

1. Atomicity(原子性)—— 同生共死

  • 定义: 事务包含的所有操作,要么全部执行成功,要么全部失败回滚(Rollback)。
  • 通俗解释: 就像原子是不可分割的一样,事务里的操作也是一个整体。
  • 案例: 转账过程中,A 扣款成功,但在 B 加钱之前系统崩了。数据库会监测到这个事务没有完成,于是自动把 A 扣掉的钱退回去(回滚),就像这件事从来没发生过一样。
  • 底层原理:Undo Log(回滚日志)实现。每做一步操作,数据库都记了个小本本,一旦失败,就反向操作把数据改回去。

代码示例(Spring):

Java

@Transactional // 开启事务
public void transfer(int fromId, int toId, double amount) {
    // 1. A 扣款
    userDao.decrementBalance(fromId, amount);
    
    // 模拟一个异常(比如除以0,或者断电)
    int i = 1 / 0; 
    
    // 2. B 加钱(由于上面报错,这行代码不会执行,且第1步会自动回滚)
    userDao.incrementBalance(toId, amount);
}

2. Consistency(一致性)—— 守规矩

  • 定义: 事务执行前后,数据库的完整性约束没有被破坏。
  • 通俗解释: 能量守恒定律。
  • 案例:
  • A 有 500 元,B 有 0 元。
  • 无论怎么转账,只要钱没转出银行系统,A 和 B 的余额总和永远应该是 500 元。
  • 如果转账完,A 剩 400,B 变成了 200,总和变 600 了,那就是破坏了一致性。
  • 另外,如果数据库规定余额不能为负数,那么 A 余额只有 100 却要转 200,事务必须失败,这也是一致性。

3. Isolation(隔离性)—— 各玩各的

  • 定义: 多个事务并发执行时,不应互相干扰。
  • 通俗解释: 你在ATM机上查余额,你老婆正好在用支付宝刷你的卡消费。这两个操作同时进行,应该互相隔离,不能让你看到的数据乱套。
  • 并发带来的问题:
  • 脏读: 你读到了别人还没提交的数据(万一他回滚了,你读的就是假数据)。
  • 不可重复读: 一个事务内两次读到的数据不一样。
  • 幻读: 一个事务内读到的行数不一样。
  • 解决办法: 数据库提供了 4 种隔离级别(Read Uncommitted, Read Committed, Repeatable Read, Serializable)来平衡性能与隔离性。MySQL InnoDB 默认使用的是 Repeatable Read(可重复读)

4. Durability(持久性)—— 落袋为安

  • 定义: 事务一旦提交(Commit),它对数据的修改就是永久的。
  • 通俗解释: 只要你看到了“交易成功”四个字,哪怕下一秒机房爆炸、服务器被雷劈了,你的钱也确确实实转过去了,数据绝不会丢失。
  • 底层原理:Redo Log(重做日志)实现。上一篇文章提到过,数据修改会先写日志。只要日志在磁盘上,重启后数据库就能根据日志重新构建数据。

总结

面试时如果你能用这个逻辑讲出来,稳过:

  • 原子性 (A):要么全做,要么全不做(靠 Undo Log)。
  • 持久性 (D):由于断电等原因,已提交的数据不能丢(靠 Redo Log)。
  • 隔离性 (I):并发事务之间互不干扰(靠 锁 和 MVCC)。
  • 一致性 (C):这是最终目的。原子性、隔离性、持久性都是为了保证数据的一致性。
相关文章
|
5月前
|
SQL 监控 druid
【性能优化】拒绝性能瓶颈!数据库连接池配置详解与调优实战
本文深入讲解数据库连接池核心原理与调优技巧,涵盖HikariCP和Druid配置要点,解析四大关键参数、黄金连接数公式及Druid监控功能,助你科学设置连接池,避免性能瓶颈。
|
5月前
|
消息中间件 缓存 NoSQL
【Redis进阶】不止是缓存!Redis的5种核心数据结构与实战场景全解析
本文深入浅出地解析了Redis五大核心数据结构:String、Hash、List、Set和ZSet,结合图解与实战场景,涵盖缓存、计数器、分布式锁、购物车、消息队列、排行榜等典型应用,助你摆脱“只会SET/GET”的困境,真正发挥Redis的高性能潜力。
|
Web App开发 移动开发 前端开发
|
6月前
|
机器学习/深度学习 SQL 关系型数据库
TRUNCATE、DELETE、DROP 的区别?
MySQL中DELETE、TRUNCATE和DROP均用于删除数据,但作用不同:DELETE删除行记录,支持WHERE条件和事务回滚,速度慢;TRUNCATE快速清空表并重置自增ID,不可回滚;DROP则彻底删除表结构与数据,操作不可逆。三者在日志记录、速度及功能上有显著差异。
754 0
|
5月前
|
关系型数据库 MySQL Nacos
CAP原理
本节介绍分布式事务中的CAP原理,即一致性(C)、可用性(A)、分区容忍性(P)三者不可兼得。分布式系统必须满足P,因此需在C与A之间权衡,选择CP或AP方案。内容结合金融、库存、订票等实际场景,解析Zookeeper、Redis、Nacos等技术的选型应用,指导如何根据业务需求合理选择分布式事务控制策略。
CAP原理
|
5月前
|
数据采集 缓存 监控
闲鱼 item_get - 商品详情接口对接全攻略:从入门到精通
闲鱼item_get接口可精准获取二手商品详情,支持商品ID查询标题、价格、成色、卖家信息等全量数据,适用于比价、运营、风控等场景。本文提供从权限申请、签名认证到生产优化的全流程对接指南,确保稳定高效调用。
|
5月前
|
SQL 存储 关系型数据库
数据库的行级锁与表锁
表锁无死锁,但并发低,读写互斥;行锁基于索引,支持高并发,但可能死锁。若SQL未走索引,行锁失效转为表锁。行锁适用于避免不可重复读,事务中增删改自动加排他锁,且不可锁定同一索引。
|
9月前
|
SQL Oracle 关系型数据库
【赵渝强老师】Oracle客户端与服务器端连接建立的过程
Oracle数据库采用客户端-服务器架构,客户端通过其网络环境与服务器通信,实现数据库访问。监听程序负责建立连接,通过命令lsnrctl可启动、停止及查看监听状态。本文介绍了监听器的作用及相关基础管理操作。
376 0
|
SQL Oracle 算法