[MySQL]事务ACID详解

2023-11-04


专栏简介 :MySql数据库从入门到进阶.

题目来源:leetcode,牛客,剑指offer.

创作目标:记录学习MySql学习历程

希望在提升自己的同时,帮助他人,,与大家一起共同进步,互相成长.

学历代表过去,能力代表现在,学习能力代表未来! 


目录 

1. 事务的概念

2. 事务的特性

3.事务控制语法

4. 事务并发异常

5. 事务隔离级别


1. 事务的概念

事务指逻辑上的一组操作, 组成这组操作的各个单元, 除非全部正确执行, 否则中间任何一个操作出现错误, 都会回滚(rollback)到数据最初的安全状态, 以保证不会对系统数据库造成错误的改动. 


2. 事务的特性

谈到事务一般都是以下四点:

  • 1.原子性(Atomicity)

过去原子被认为是分割的最小单位, 类比事务, 事务是一个不可分割的工作单位, 事务中的操作要么全部成功要么全部失败.

 这个过程包含两个步骤:

1.A转给B 200元, A剩余600元

2.B收到A 200元, B剩余400元

原子性表示, 这两个步骤不可分割, 要么全部成功要么全部失败.


  • 2.一致性(Consistency)

事物提交前和提交后, 数据都是合法的一致性状态, 即使发生异常也不会破坏数据库的完整性约束.(符合逻辑运算)

假设要求事务执行前后总有A的账户余额+B的账户余额=1000, 那么事务执行后该完整性约束仍然满足.因此这个事务就满足一致性.


  • 3.隔离性(isolation)

多个用户并发(同时)访问数据库时, 数据库为每一个用户都开启事物, 为了防止多个事物的操作数据相互干扰, 多个并发事物之间要相互隔离.隔离性通过事务的隔离级别来定义, 并用锁机制来保证写操作的隔离性, MVCC来保证读操作的隔离性.

 假设A只有500元, 如果没有隔离性, 两个事务同时进行, 事务2可能会读取到事务1还未提交的数据.此时A就会转出700, 这显然是不符合逻辑的, 隔离性就是为了解决上述问题.


  • 4.持久性(durability)

一个事物一旦被提交, 它对数据库的改变就是永久的, 不管是断电还是机器故障也不会对其有任何影响, 通过事务中重做日志(redo log)来保证, 事务修改之前会将变更信息预写到 redo log中, 如果数据库宕机, 恢复后会读取redo log中的数据恢复数据.


3.事务控制语法

事务控制有几个重要节点, 分别是事务的开启, 提交, 回滚保存.

1.开启事务代表事务开始执行, 语句为:

start transaction --或者
begin

2.提交事务代表将事务所有的更新内容都提交到磁盘的物理数据库, 语句为:

commit

3.如果发生异常需要回滚, 语句为:

rollback

Tips:事务一但提交就不能回滚, 异常在代码执行过程中发生异常需要执行的是rollback而不是commit

示例:

张三向李四转100元

--事务正常执行, 提交.
begin; --开启事务
update account set money = money-100.00 where account.name = "张三";
update account set money = money+100.00 where account.name = "李四";
commit; --提交事务

--事务发生异常, 回滚.
begin; --开启事务
update account set money = money-100.00 where account.name = "张三";
update account set money = money+100.00 where account.name = "李四";
rollback; --事务回滚

在复杂场景中, 我们不需要全盘回滚, 而是分批执行回滚到某个节点就好, 相当于一个大事务低下嵌套多个子事务.常用savepoint来实现.

begin;
insert into students (id) values (1);
savepoint s1;
insert into stduents (id) values (2);
rollback to s1;--回滚到保留点s1, 因此1成功写入, 2被回滚, 最终结果为1.

--释放保留点
release savepoint s1;

4. 事务并发异常

实际生产过程中, 经常会遇到多事务并发访问的情况, 如果没有设置恰当的事物隔离级别, 就可能导致一些异常的情况出现, 最常见的几种异常为:脏读(Dirty Read) ,幻读(Phantom Read) 和不可重复读(Unrepeatable Read).

  • 1. 脏读

脏读指一个事务访问到了另一个事务未提交的数据.

示例:

1.假设张三有5000元的工资,事务将他的工资改为8000 此时事务A还未提交.

2.这时事务B读取到张三的工资为8000元.

3.随后事务A发生了异常, 回滚了事务, 张三的工资又回滚到5000.

4.事务B读取到的工资8000为脏数据.事务B做了异常脏读.

解决方式:

Read Committed 读已提交, 将事务的隔离级别设置为读已提交, 事务就不会访问到脏数据.


  • 2. 不可重复读

不可重复读指一个事务多次读取同一个数据的过程中, 数据值发生改变, 导致没法读到相同的数据,多指表中同一条数据 update 现象.

示例:

1.假设事务A程序员拿着5000块钱工资去消费.

2.当他准备付款时事务A开启, 收费系统检测到他卡里有5000元.

3.就在此时, 事务B程序员妻子用将5000元全部转入她的银行卡里, 并提交.

4.当收费系统准备扣钱时, 发现卡里已经没钱了.程序有就很纳闷, 卡里明明是有钱的...

解决方式:

Repeatable Read 可重复读, 当读取事务开启时, 不允许进行修改操作. 

此时当事务A程序员开启付款事务时, 不允许其他事务的修改操作(Update), 这时她的妻子将不能转出金额.收费系统就可以正常扣款了.

Tips:由此可以得出, 不可重复读对应的是修改(Update)操作, 但仍然可能发生幻读, 幻读对应的是insert插入操作.


  • 3. 幻读

幻读指一个事务多次读取同一个数据的过程中,  全局数据(表的结构)发生了改变, 仿佛产生了幻觉, 多指表的 insert/delete 现象.

示例:

1.假设事务A妻子查看程序员的消费账单(全盘扫描), 发现只花了2000元.

2.此时事务B程序员花了10000元买了一台电脑.及insert新增了一条消费记录并提交.

3.当事务A妻子打印程序员消费记录时(妻子事务A提交).

4.发现花了12000似乎产生了幻觉.

解决方式:

Serializable是最高的事物隔离级别, 在该级别下事务串行化执行, 可以避免所有异常, 但数据库效能低下, 一般不建议使用.


5. 事务隔离级别

虽然串行化的事物处理最安全, 但无法满足数据库高并发访问的需求, 因此不得不在可控的范围内降低数据库的隔离标准来换取事务的并发能力.

Tips:事务的隔离级别和数据库的并发性是成反比的.隔离级别越低, 并发性越高.

数据库有四种隔离标准, 分别是:读未提交(Read Uncommitted),  读已提交(Read Committed), 可重复读(Repeatable Read), 串行化(Serializable)


  • 1.读未提交(Read Uncommitted)

允许读取未提交的内容, 这种级别下的查询不会加锁, 因此脏读, 幻读, 不可重复读都有可能发生.


  • 2.读已提交(Read Committed)

只允许读取已提交的内容, 这种情况下不会发生脏读, 因为脏数据属于为提交的数据, 但幻读和不可重复读依然有可能发生.


  • 3.可重复读(Repeatable Read) MySQL的默认隔离级别

用行级锁来保证一个事务在相同查询条件下两次查询结果一致 , 可以避免脏读, 不可重复读, 但无法避免幻读.


  • 4.串行化(Serializable)

用表级锁来保证所有事务串行化, 可以防止所有的异常情况, 但牺牲了数据库的并发性.


四种隔离级别对上述三种异常的容忍度如下:(✅表示可容忍 ❌表示禁止)

事务隔离级别 脏读 不可重复读 幻读
读未提交 Read Uncommitted
读已提交 Read Committed
可重复读 Repeatable Read
串行化 Serializable

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

[MySQL]事务ACID详解 的相关文章

  • Java 的 System.arraycopy() 对于小数组有效吗?

    是Java的System arraycopy 对于小数组来说是高效的 或者它是本机方法这一事实是否使其可能比简单的循环和函数调用效率低得多 本机方法是否会因跨越某种 Java 系统桥梁而产生额外的性能开销 稍微扩展一下 Sid 所写的内容
  • 无法让远程 EJB 与 Wildfly 上的 EJB 客户端 API 配合使用

    我目前正在努力让远程 EJB 调用在 wildfly 8 x 和 9 x 上工作 详细来说 它是关于使用 EJB 客户端 API 方法从独立客户端应用程序 而不是从另一个应用程序服务器 进行远程调用 远程命名方法适用于我 但不适用于我的场景
  • Spring Rest POST Json RequestBody 不支持内容类型

    当我尝试使用 post 方法发布新对象时 RequestBody 无法识别 contentType Spring 已经配置完毕 POST 可以与其他对象一起使用 但不能与这个特定对象一起使用 org springframework web
  • 使用 Hibernate 或 Spring 打印 DBMS_OUTPUT.put_line

    我想知道 Hibernate 或 Spring 或任何第 3 方库是否提供将 DBMS OUTPUT put line 消息直接打印到 system out 或日志文件的能力 目的是在控制台中同时显示 PLSQL 日志消息和 java 日志
  • 嵌套枚举是静态的吗?

    读书时这个问题 https stackoverflow com questions 25011061 why can enum implementations not access private fields in the enum cl
  • 为什么这不会导致 NullPointerException?

    public class Null public static void greet System out println Hello world public static void main String args Null null
  • Tomcat JDBC 池中没有足够的空闲连接

    给定以下 Tomcat JDBC 连接设置
  • Ubuntu 的打包 - Web 应用程序

    Web 应用程序没有与 C 或类似文件不同的 make 文件 但是 它需要放置在特定的目录中 例如 var www 我是 Linux 打包新手 所以我的问题是 如何将我的应用程序打包到 deb 中 以便在安装时将其放入 etc myprog
  • 创建一个 int 类型的随机数组。爪哇

    我需要创建一个随机的 int 数组 并按我自己的类对其进行排序 这是我制作数组的地方 public class MyProgram9 public static void main String args int list new int
  • 使用java读取Excel工作表的单列

    我有一张 Excel 表格 我想编写一个方法 该方法将参数作为要读取的列号 并返回一个由该列中的所有数据组成的数组 然后将该列元素放置在 xml 工作表中 我怎样才能编写一个方法来做到这一点 使用 Apache POI 您可以在他们的使用页
  • Mysql:将数据库从本地时间转换为UTC

    我需要从本地时间 ut UTC 转换现有的 日期时间字段 数据库 这些值和日期时间存储在时区为 CET 1 夏令时 2 的服务器上 选择我使用的数据时UNIX TIMESTAMP 它神奇地补偿了一切 即时区偏移和夏令时 如果我正确阅读了文档
  • 捕获 XSS(跨站脚本)攻击的最佳正则表达式(Java 中)?

    杰夫实际上在净化 HTML http refactormycode com codes 333 sanitize html 但他的示例是用 C 编写的 而我实际上对 Java 版本更感兴趣 有人有更好的 Java 版本吗 他的示例是否足以直
  • 使用 Mockitos 传递参数化输入

    我正在使用 Mockito 进行单元测试 我想知道是否可以使用 Junit 测试中的方式发送参数化输入参数 e g InjectMocks MockClass mockClass new MockClass Test public void
  • 在同一台计算机上设置 JBoss 的多个实例

    我在 JBoss 社区中找到了下一页 http www jboss org community wiki ConfigurePorts http www jboss org community wiki ConfigurePorts 有下一
  • 如何强制初始化 Hibernate JPA 代理以在 JSON 调用中使用它

    我有一个 Spring 3 JPA 2 0 应用程序 在我的 Controller我需要一个初始化的对象 但我有代理 我需要能够以编程方式初始化它 我需要类似的功能org hibernate Hibernate initialize Obj
  • Java中如何限制文件大小

    我正在我的应用程序中创建一个文件 并继续向该文件中写入一些内容 但是当我的文件达到一定大小 比如说 100 行 后 我想删除第一行并将新行写入底部 要求是我的文件应该受到限制 但它应该保留我写入文件的最新内容 请告诉我在Java中是否可行
  • Google App Engine数据存储区字符串编码问题

    你好 我正在使用 Google App Engine 进行一个项目 并且需要存储一些字符串 我正在使用 Java 和JDOHelper getPersistenceManagerFactory transactions optional 在
  • 使用 Tomcat 和 gradle 进行休眠

    免责声明 我是 Java 新手 我正在尝试使用 Tomcat 和 Gradle 设置 Hibernate 构建运行正确 但看起来像persistence xml文件未被读取 我的项目结构如下 build gradle src main ja
  • MySQL - 连接 a 或 b

    假设我有一个TABLE a其中一个COLUMN data是一个join其他 2 张桌子 TABLE b and TABLE c 因为我想得到一个COLUMN info in b or c 事情是a data将匹配only with b da
  • 错误:运算符不存在:整数 = 字符变化,使用 Postgres 8.2

    我有一个用旧版本的 Eclipse Ganymede 如果我没记错的话 开发的 Java EE Web 应用程序 我最近迁移到 Kubuntu 12 04 LTS 并将应用程序迁移到 Eclipse Kepler 我从 Eclipse 网站

随机推荐

  • 【论文笔记】RA-UNet: A Hybrid DeepAttention-Aware Network to ExtractLiver and Tumor in CT Scans

    RA UNet 用于在CT扫描中提取肝脏和肿瘤的一种混合深度注意力感知网络 论文地址 https arxiv org abs 1811 01328 代码地址 https github com RanSuLab RAUNet tumor se
  • Vue实现倒计时组件(可自定义时间倒计时功能的组件):

    一 创建countDown vue 倒计时组件
  • 【genius_platform软件平台开发】第九十四讲:int64_t的格式化问题(lld和PRId64)

    问题起因是在进行上位机软件优化的工作安排时 同事对unsigned long long 类型的时间戳进行了格式化输出优化 从 ull优化为了 PRIu64 我进行代码合并请求处理的时候突然感觉这个可以仔细查一下 查阅到的相关资料如下 1 i
  • msrcr算法matlab代码_图像处理之Retinex增强算法(SSR、MSR、MSRCR)

    视网膜 大脑皮层 Retinex 理论认为世界是无色的 人眼看到的世界是光与物质相互作用的结果 也就是说 映射到人眼中的图像和光的长波 R 中波 G 短波 B 以及物体的反射性质有关 其中I是人眼中看到的图像 R是物体的反射分量 L是环境光
  • MacOS编译NGINX1.17

    背景 使用Mac的开发者大多数的时候通过 brew 命令安装各类软件 比如 PHP Python Nodejs Nginx 也不例外 默认情况下Nginx安装在 usr local Cellar nginx 通过brew link ngin
  • ImportError: ERROR: recursion is detected during loading of “cv2“ binary extensions. Check OpenCV in

    1 import cv2错误 ImportError ERROR recursion is detected during loading of cv2 binary extensions Check OpenCV installation
  • C/C++语言图形化窗口创建,与设置。

    1 首先我们要引入头文件 include
  • log4j2远程代码执行漏洞原理与漏洞复现(基于vulhub,保姆级的详细教程)

    漏洞原理 啥是log4j2 log4j2是apache下的java应用常见的开源日志库 是一个就Java的日志记录工具 在log4j框架的基础上进行了改进 并引入了丰富的特性 可以控制日志信息输送的目的地为控制台 文件 GUI组建等 被应用
  • pybind11的安装以及库的使用(ubuntu20.14)

    目录 1 安装第三方库 2 安装pybind11 1 源码下载 2 编译安装 1 安装第三方库 sudo apt get install cmake 3 18 2 已经安装cmake 升级一下 sudo pip install upgrad
  • Matlab数据统计与分析(一)

    本文主要讲述matlab在统计分析中的一些简单应用 后续还将推出多项式计算 数据插值及其应用 曲线拟合及其应用 敬请期待 一 求最大元素与最小元素 max 求向量或矩阵的最大元素 min 求向量或矩阵的最小元素 调用格式如下 1 y max
  • 单细胞专题

    单细胞测序的概念 上节我们讲到转录组测序相关内容 这期将继续学习单细胞转录组测序 单细胞测序技术 single cell sequencing 简单来说 就是在单个细胞水平上 对基因组 转录组及表观基因组进行测序分析的技术 图1 图1 单细
  • 借助云开发搭建专属技术博客微信小程序(附源码)

    导语 一直对小程序开发很感兴趣 之前就准备做一款属于自己的小程序 无奈还需要购买云服务器和部署后台 有点麻烦 自从知道有了云开发这个免去服务器搭建和运维的一站式后端云服务 神器 就一鼓作气花了几个周末的时间做了一款自己的博客小程序 如果你也
  • ES写入性能的提升方案

    从社区上看到一个问题 回答了一下 总结了下提高写入性能的方案 服务端 1 refresh refresh interval 1s indices memory index buffer size 10 2 translog Index tr
  • vSphere使用技巧:自定义管理规范使用

    本文转载至 http blog chinaunix net uid 21089721 id 1565198 html 一 前言 我们在使用vCenter Client的时候 常常会使用模板来安装服务器 但是这样就会带来一些问题 例如WIND
  • Windows通过某端口号找对应的进程窗口句柄

    以下为 Qt通过netstat exe程序获取相关信息 通过端口号获取进程ID 的姊妹篇 通过端口号找对应的窗口句柄 以上所提及的API 均为Win32的系统API 不涉及Qt的相关方法
  • Shell Sort 希尔排序

    希尔排序 Shell Sort 又叫做缩小增量排序 diminishing increment sort 是一种很优秀的排序法 算法本身不难理解 也很容易实现 而且它的速度很快 插入排序 Insertion Sort 的一个重要的特点是 如
  • Oracle中没有 if exists(...)

    对于Oracle中没有 if exists 的语法 目前有许多种解决方法 这里先分析常用的三种 推荐使用最后一种 第一种是最常用的 判断count 的值是否为零 如下 declare v cnt number begin select co
  • 如何搭建自己的服务器机房

    1 地下室这种环境 铲成毛坯先做一遍防水防潮抗震套餐 这个便宜 相对 2 找国家电网拉两个不一样变电站的电线杆子来做380V 之前做过一个机房的这种需求 也就二百来万吧 3 每个运营商 不同机房 做两个方向两个井 两条8 16芯缆 运营商大
  • HTML5 详细介绍 及应用实例

    HTML5 概况 什么是 HTML5 HTML 5有两大特点 首先 强化了 Web 网页的表现性能 其次 追加了本地数据库等 Web 应用的功能 HTML 5是近十年来Web开发标准最巨大的飞跃 和以前的版本不同 HTML 5并非仅仅用来表
  • [MySQL]事务ACID详解

    专栏简介 MySql数据库从入门到进阶 题目来源 leetcode 牛客 剑指offer 创作目标 记录学习MySql学习历程 希望在提升自己的同时 帮助他人 与大家一起共同进步 互相成长 学历代表过去 能力代表现在 学习能力代表未来 目录