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

本文涉及的产品
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):这是最终目的。原子性、隔离性、持久性都是为了保证数据的一致性。
相关文章
|
6天前
|
Java Linux 开发工具
Linux
本文介绍如何将一个简单的SpringBoot应用打包并部署到Linux服务器。包括项目搭建、JAR包打包、JDK安装配置、应用上传与启动,以及通过心跳接口验证服务是否正常运行的完整流程,适用于Java应用的Linux部署入门学习。
|
5天前
|
监控 Java Sentinel
Sentinel安装与集成
介绍如何切换hmall-micro项目至dev_02分支并提交代码,强调多分支并行开发模式。随后讲解Sentinel服务保护框架的安装与配置,包括控制台部署、本地运行及项目集成,实现微服务熔断降级,并通过簇点链路监控接口。
Sentinel安装与集成
|
5天前
|
人工智能 Java 微服务
微服务保护方案
Spring Cloud微服务中,服务保护机制对保障系统稳定性至关重要,主要包括熔断、降级、超时、线程隔离和限流。熔断快速失败避免雪崩,降级提供默认响应保证核心功能,超时防止长时间等待,线程隔离限制故障影响范围,限流控制流量峰值,共同提升系统容错与可用性。
微服务保护方案
|
5天前
|
缓存 前端开发 安全
数据同步原理
Soul网关通过推拉模式实现配置数据同步,支持WebSocket、HTTP长轮询和Zookeeper三种策略。管理员在后台变更配置后,事件被发布并根据同步策略推送到网关,实现秒级更新。HTTP长轮询借鉴Apollo与Nacos设计,结合异步Servlet机制,确保准实时且高效。
数据同步原理
|
5天前
|
Java Sentinel 微服务
实现降级
本文介绍如何在Spring Cloud微服务中通过Sentinel实现Feign接口的降级处理。重点采用`FallbackFactory`方式,在调用方(如cart-service)为`ItemClient`接口编写降级逻辑,捕获远程调用异常并返回兜底数据。需配置Feign启用Sentinel,定义降级类实现`FallbackFactory`接口,并在`@FeignClient`中指定`fallbackFactory`。最终通过停止item-service进行测试,验证购物车页面在商品信息获取失败时仍可降级展示,保障系统稳定性。适用于服务熔断与容错场景。
实现降级
|
5天前
|
存储 算法 BI
xxljob本地运行
本文介绍XXL-JOB分布式任务调度框架的部署与使用,涵盖源码获取、服务端数据库配置、客户端注册及任务调度配置,支持多种路由策略与分片广播,助力高效实现定时任务管理。
 xxljob本地运行
|
5天前
|
弹性计算 运维 监控
【运维排查】服务器CPU飙升100%?别慌,教你3步精准定位“罪魁祸首”
当服务器CPU飙高时,别急着重启!本文教你四步精准排查:用`top`定位高占用进程,`top -Hp`找出耗CPU线程,`printf`转十六进制,再通过`jstack`结合线程ID定位到具体代码行。快速锁定死循环、频繁GC或复杂计算等问题根源,成为团队中的故障排查高手。
|
5天前
|
运维 Shell 应用服务中间件
【速查手册】Docker常用命令大全:这20%的命令解决了80%的问题
本文精炼总结 Docker 高频命令,按镜像管理、容器生命周期、排查调试、清理维护四大场景分类,详解常用参数与实战示例,附速查表,助你高效掌握核心操作,提升开发运维效率。
|
5天前
|
缓存 Java Docker
【Docker实战】如何写出“性感”的Dockerfile?从1GB瘦身到100MB的秘籍
本文介绍如何编写高效、安全的Dockerfile,以Java和Python为例,分享四大核心技巧:多阶段构建减小镜像体积,利用缓存加速构建,选用轻量基础镜像,配置.dockerignore忽略无用文件。助你打造小巧、快速、安全的容器镜像,提升部署效率与安全性。
|
5天前
|
存储 运维 Java
【Docker入门】5分钟彻底搞懂镜像、容器与仓库:Docker的核心三剑客
Docker 通过“镜像、容器、仓库”三大核心实现“一次构建,到处运行”。镜像如食谱,容器是做好的菜,仓库似超市货架。用生活化比喻秒懂其原理与协作流程。