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

本文涉及的产品
RDS AI 助手,专业版
RDS MySQL DuckDB 分析主实例,基础系列 4核8GB
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):这是最终目的。原子性、隔离性、持久性都是为了保证数据的一致性。
相关文章
|
3月前
|
SQL 监控 druid
【性能优化】拒绝性能瓶颈!数据库连接池配置详解与调优实战
本文深入讲解数据库连接池核心原理与调优技巧,涵盖HikariCP和Druid配置要点,解析四大关键参数、黄金连接数公式及Druid监控功能,助你科学设置连接池,避免性能瓶颈。
|
Web App开发 移动开发 前端开发
|
4月前
|
机器学习/深度学习 SQL 关系型数据库
TRUNCATE、DELETE、DROP 的区别?
MySQL中DELETE、TRUNCATE和DROP均用于删除数据,但作用不同:DELETE删除行记录,支持WHERE条件和事务回滚,速度慢;TRUNCATE快速清空表并重置自增ID,不可回滚;DROP则彻底删除表结构与数据,操作不可逆。三者在日志记录、速度及功能上有显著差异。
525 0
|
3月前
|
存储 关系型数据库 索引
聚簇索引及其优缺点
聚簇索引是一种数据存储方式,InnoDB通过主键构建B+树组织数据,叶子节点即数据页。若无主键,则选非空唯一索引或隐式创建主键。辅助索引(二级索引)需两次查找:先查主键值,再查数据行。优点是查询快,尤其主键排序与范围查询;缺点是插入依赖顺序,更新主键代价高,且易引发页分裂。
|
4月前
|
JSON 安全 JavaScript
深入浅出解析 HTTPS 原理
HTTPS是HTTP与SSL/TLS结合的安全协议,通过数字证书验证身份,利用非对称加密安全交换会话密钥,再以对称加密高效传输数据,确保通信的机密性、完整性和真实性。整个过程如同建立一条加密隧道,保障网络交互安全。
2230 16
|
7月前
|
SQL Oracle 关系型数据库
【赵渝强老师】Oracle客户端与服务器端连接建立的过程
Oracle数据库采用客户端-服务器架构,客户端通过其网络环境与服务器通信,实现数据库访问。监听程序负责建立连接,通过命令lsnrctl可启动、停止及查看监听状态。本文介绍了监听器的作用及相关基础管理操作。
309 0
|
9月前
|
SQL 关系型数据库 MySQL
数据控制语言 DCL
本篇文章详细介绍了 MySQL 中 DCL(数据控制语言)的核心命令,包括用户账户的创建、修改、删除 (CREATE USER, ALTER USER, DROP USER),以及数据库权限的授予与撤销 (GRANT, REVOKE)。文章通过清晰语法和实际示例,指导读者如何设置强密码、限制登录主机、精细授权数据库对象,帮助构建更安全可控的数据库系统。结合最佳实践与练习题,本教程适合 DBA 入门者和开发人员快速掌握 MySQL 权限管理能力。
349 1
|
9月前
|
SQL 关系型数据库 MySQL
MySQL 常用函数
我们这次全面梳理 MySQL 中的常用函数,涵盖 聚合函数、字符串函数、日期时间函数、数学函数 和 控制流函数 等五大类。每类函数均配有语法说明与实用示例,帮助读者提升数据处理能力,如统计分析、文本处理、日期计算、条件判断等。文章结尾提供了丰富的实战练习,帮助读者巩固和应用函数技巧,是进阶 SQL 编程与数据分析的实用工具手册。
607 2
|
存储 Windows
在 PowerShell 中获取代理设置
【8月更文挑战第27天】
1214 6
|
SQL Oracle 算法