一篇文章搞懂MySQL的事务与隔离级别

事务

概述

一个事务其实就是一个完整的业务逻辑,是一个最小的工作单元。要么同时成功,要么同时失败,不可再分

假设转账,从A账户向B账户转账10000
A账户的钱减去10000(update语句)
B账户的钱加上10000(update语句)
这就是一个完整的业务逻辑

这两个update语句要求必须同时成功或者同时失败,才能保证钱是正确的。
只有DML(insert、delete、update)语句才会有事务这一说,其他语句和事务无关。
一旦你的操作涉及到增、删、改,那么就一定要考虑安全问题。

事务的实现与操作

主要是通过redo和undo来保证的事务的一致和回滚等操作
redo和undo的区别

  • redo和undo的作用都可以视为是一种恢复操作
    • redo恢复提交事务修改的页操作
    • 而undo回滚行记录到某个特定版本
  • 因此两者记录的内容不同
    • redo通常是物理日志,记录的是页的物理修改操作。
    • undo是逻辑日志,根据每行记录进行记录
redo
  • 重做日志是为了实现事务的持久性的,由redo log buffer 和redo log file组成
  • InnoDB通过Force Log at Commit机制实现事务的持久性
    • 事务的日志写入redo文件进行持久化,等待commit
  • 重做日志在InnoDB中,由redo log和undo log组成
    • redo log用来保证事务的持久性,是顺序写的
    • undo log用来帮助事务回滚及MVCC的功能,随机读写
  • InnoDB确保每次日志写入重做日志文件中后要调用fsync来保证数据写入到了磁盘
    • fsync的效率取决于磁盘的性能=>决定了事务提交的性能=>数据库的性能
  • InnoDB允许事务提交后不立马写入文件,而是一个周期一个周期写入
    • 显著提高性能
    • 会丢失事务
操作

提交和回滚实现
提交事务:commit;
回滚事务:rollback;(回滚是回滚到上一次的提交点)
事务对应的英文单词是:transaction
mysql中默认事务是自动提交的,就是每执行一条DML语句,则提交一次。

  • 这种自动提交实际上是不符合我们的开发习惯,因为一个业务通常是需要多条DL语句共同执行才能完成的,为了保证数据的安全,必须要求同时成功之后再提交,所以不能执行一条就提交一条。

开启事务,mysql就不会默认事务是自动提交了,会关闭自动提交机制

start transaction;//开启事务,并且关闭自动提交机制 
insert into dept_bak values(10, 'abc', 'tj');
rollback; //回滚,则上面的insert语句被撤销start transaction;//开启事务,并且关闭自动提交机制 
insert into dept_bak values(10, 'abc', 'tj');
commit;//提交事务rollback;//回到上一个cimmit后面

事务的分类

  • 扁平事务
    • 所有操作都处于同一层次,由BEGIN WORK开始,由COMMIT WORK/ROLLBACK WORK结束
    • 操作是原子的,要么都执行,要么都回滚
    • 不能提交或者回滚事务的某一部分,或分几个步骤提交
  • 带保存点的扁平事务
    • 支持扁平事务支持的操作
    • 允许在事务执行过程中回滚到同一事务中较早的一个状态
    • 保存点在事务内部是递增,能回滚到任意正确的保存点
    • 保存点是非持久化的
  • 链事务
    • 提交一个事务时,释放不需要的数据对象,将必要的处理上下文隐式地传给下一个要开始的事务
    • 只能恢复到最近一个的保存点
  • 嵌套事务
    • 层次结构框架,由一个顶层事务(top-level transaction)控制着各个层次的事务
    • 由若干事务组成的一棵树,子树既可以是嵌套事务,也可以是扁平事务
    • 处在叶节点的事务是扁平事务。但是每个子事务从根到叶节点的距离可以是不同的。
    • 根节点的事务称为顶层事务,其他事务称为子事务,事务的前驱称为父事务,事务的下一层称为儿子事务
    • 任何子事务都在顶层事务提交后才真正的提交,任意一个事务的回滚会引起它的所有子事务一同回滚
  • 分布式事务
    • 是一个在分布式环境下运行的扁平事务,因此需要根据数据所在位置访问网络中的不同节点

事务的特性

ACID
原子性(A)

  • 说明事务是最小的工作单元,不可再分。

一致性(C)

  • 所有事务要求,在同一个事务当中,所有操作必须同时成功,或者同时失败,以保证数据的一致性。
  • 一致性也是指符合我们的业务逻辑和结果

隔离性(I)

  • A事务和B事务之间具有一定的隔离。相当于多线程并发访问的每个线程。

持久性(D)

  • 事务最终结束的一个保障。事务提交,就相当于将没有保存到硬盘上的数据保存到硬盘上!

事务的隔离性

形象的说,A教室和B教室中间有一道墙,这道墙可以很厚,也可以很薄。越厚,隔离级别越高。

事务与事务之间有4个隔离级别

读未提交:read uncommitted(最低隔离级别)《没有提交就能读到》

  • 事务A可以读取到事务B未提交的数据。(没有提交就读取到了)
  • 这种隔离级别存在的问题就是:脏读现象(Dirty Read),我们称堵到了脏数据。
  • 这种隔离级别一般都是理论上的,大多数的数据库隔离级别都是二挡起步。

读已提交:read commited《提交后才能读到》

  • 事务A只能读取到事务B提交之后的数据。解决了脏读现象。
  • 这种隔离级别存在的问题是:不可重复读取数据。
    • 什么是不可重复读取数据呢?在事务开启之后,第一次读到的数据是3条,当前事务还没有结束,可能第二次再读取的时候,读到的数据是4条,3不等于4称为不可重复读取。
  • 这种隔离级别是比较真实的数据,每次读到的数据是绝对真实的。
  • Oracle数据库默认的隔离级别是:read commited

可重复读:repeatable read《提交后也读不到,读到的都是本事务刚开始的数据》

  • 当前事务开始事务的时候,我备份了一份,我读的时备份的数据。不管你外界的事务有没有跟新数据,对我没影响。
  • 提交之后也读不到,永远读取的都是刚开启事务时的数据。
  • 事务A开启之后,不管多久,每一次在事务A中读取到的数据,都是一致的。即使事务B将数据已经修改,并且提交了,事务A读取到的数据还是没有发生改变,这就是可重复读。
  • 存在的问题:可能会出现幻影读。每一次读取到的数据都是幻影,不真实。
  • mysql中默认的事务隔离级别。

序列化/串行化:serializable(最高的隔离级别)

  • 这是最高隔离级别,效率最低。解决了所有的问题。
  • 这种隔离级别表示事务排队,不能并发!
  • synchronized,线程同步(事务同步)
  • 每次读取到的数据都是最真实的,但是效率是最低的。

验证事务隔离级别

查看隔离级别:select @@tx_isolation

验证:read uncommitted

设置全局隔离级别为“读未提交”:set global transaction isolation level read uncommitted;
8ba0c36aa07e40f1a01bb9c9271110f2.png解释:打开两个DOS窗口,分别进入数据库,一个DOS窗口开启一个事务。开启事务后,事务B中进行的操作还未提交,在事务A中就能查询到。
个人理解:读未提交,A可以读到B未提交的数据,也就是当A开启事务,B也开启事务后,事务B去修改了数据,但是B并没有commit提交,同时A也没有提交,此时A查询数据,能够读到修改了的数据
此时A没提交,B也没提交就能读到数据

验证:read commited

设置全局隔离级别为“读已提交”:set global transaction isolation level read committed;
607788cc14ea47e6a0d8a083c2e78653.png
解释:事务B进行操作未提交,在事务A查询不到修改后的记录。只有事务B提交了,才能在事务A查询到。(事务B提交,事务A才能查到)
个人理解:读未提交,A可以读到B提交后的数据,也就是当A开启事务,B也开启事务后,事务B去修改了数据,但是B并没有commit提交,同时A也没有提交,此时A查询的数据是没有修改的数据,只有当Bcommit提交事务后,A去读就能读到修改后的数据.
此时A未提交,B提交就能读到数据

验证:repeatable read

设置全局隔离级别为“可重复读”:set global transaction isolation level repeatable read;
6dc91b7ac8134e4eb59e9a28c838c62b.png
解释:事务B操作完提交后,在事务A中查询到的数据,依然是开启事务之前的数据。不论事务B做什么操作,提交几次,只要事务A没提交,查询到的就是以前的。(事务B提交,事务A提交了才能查到)
个人理解:读未提交,A提交后才可以可以读到B提交后的数据,也就是当A开启事务,B也开启事务后,事务B去修改了数据,Bcommit提交了,但是此时A没有提交,这时A查询的数据是没有修改的数据,即使当Bcommit提交事务,A读到的数据依旧是A开启事务之前的数据.只有当A提交事务后,再去查询才能查询到B修改的数据
此时A提交,B提交就能读到数据

验证:serializable

设置全局隔离级别为“序列化”:set global transaction isolation level serializable;
aa41ffcfca87462fae615c443e7bfdda.png
解释:在事务A中进行一些操作,没提交,事务B就去查询结果,会卡在那里。当事务A提交,事务B会立马出现结果。相当于同步了。
个人理解:如果A开启事务后,对user表进行了修改,B去查询user表就会卡主,类似于synchronized锁住了user表,但是当B去查询另外的表就不会卡住.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/49094.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【HarmonyOS学习】用户文件访问

概述 文件所有者为登录到该终端设备的用户,包括用户私有的图片、视频、音频、文档等。 应用对用户文件的创建、访问、删除等行为,需要提前获取用户授权,或由用户操作完成。 用户文件访问框架 是一套提供给开发者访问和管理用户文件的基础框…

Web开发:属性

属性 介绍全局属性特定元素属性超链接元素 (< a >)图像元素 (< img >)输入元素 (< input >)表单元素 (< form >)表格元素 (< table >) 其他元素段落元素 (< p >)列表元素 (< ul >, < ol >, < li >) 新增的HTML5属性示例…

无需抠图!AI绘画直接文本生成透明底图层,设计师必看的ComfyUI透明图层生成工作流教程!(附插件模型)

大家好&#xff0c;我是画画的小强 AI 绘画自出现以来一直都在不断发展完善&#xff0c;实现了很多我们在实际应用中迫切需要的功能&#xff0c;比如生成正确的手指、指定的姿势、准确的文本内容等。上周&#xff0c;又一个重磅新功能在开源的 SD 生态内实现了——直接通过文本…

PyCharm中的版本控制大师:Git的高效应用

PyCharm中的版本控制大师&#xff1a;Git的高效应用 在软件开发中&#xff0c;版本控制是一个不可或缺的工具&#xff0c;它帮助开发者管理代码变更、跟踪历史和协同工作。Git是目前最流行的版本控制系统之一&#xff0c;而PyCharm&#xff0c;作为一款功能强大的集成开发环境…

【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第四十四章 注册字符设备号

i.MX8MM处理器采用了先进的14LPCFinFET工艺&#xff0c;提供更快的速度和更高的电源效率;四核Cortex-A53&#xff0c;单核Cortex-M4&#xff0c;多达五个内核 &#xff0c;主频高达1.8GHz&#xff0c;2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT…

Springboot项目打包成镜像、使用docker-compose启动

Springboot项目打包成镜像、使用docker-compose启动 1、创建一个boot项目 1、添加依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSch…

什么是保费保额,什么是豁免条款

保额1000万和300万的保费具体数额会因多种因素而异&#xff0c;包括保险公司的不同、保险产品的类型、被保险人的年龄、性别、健康状况、生活习惯&#xff08;如是否吸烟&#xff09;等。因此&#xff0c;没有统一的保费标准。 保费的计算通常涉及以下几个因素&#xff1a; 风…

gitee的怎么上传项目

前提 1.先下载Git Bash (如果没有下载的宝子们下载连接如下: 链接: link ) 项目上传到Gitee步骤 1.在Gitee上建立远程仓库 2.填写相关信息 3.进入本地你想要上传的文件目录下&#xff0c;右键单击空白处&#xff0c;点击Git Bash Here 4.配置你的用户名和邮箱 git con…

【leetcode】排列序列

给出集合 [1,2,3,...,n]&#xff0c;其所有元素共有 n! 种排列。 按大小顺序列出所有排列情况&#xff0c;并一一标记&#xff0c;当 n 3 时, 所有排列如下&#xff1a; "123""132""213""231""312""321" 给定…

最简单的typora+gitee+picgo配置图床

typoragiteepicgo图床 你是否因为管理图片而感到头大&#xff1f;是时候了解一下 Typora、Gitee 和 PicGo 这个超级三剑客了&#xff0c;它们可以帮你轻松打造自己的图床&#xff0c;让你的博客图片管理变得简单又有趣。让我们开始这场神奇的图床之旅吧&#xff01; Typora …

20-c语言main函数参数`argc` 和 `argv[]` 解析

argc 和 argv[] 解析 argc 和 argv[] 是 main 函数的参数&#xff0c;用于处理命令行参数。 一、 示例命令行调用 ./a.out 123 345解释&#xff1a; ./a.out 是程序名&#xff0c;也是第一个参数。123 和 345 是运行时传递的额外参数。 二、main 函数定义 int main(int a…

7.20 模拟赛总结 [邻项交换] + [决策单调性]

只放题解喽 题解 T1T2T3T4 T1 等价于维护差分数组&#xff0c;数据范围较小&#xff0c;map 套 vector 维护即可 更大的数据范围可以 hash 做 T2 神奇贪心 本题关键在于定序&#xff0c;考虑顺序确定后答案怎么求 设 f i f_i fi​ 表示 第 i i i 件衣服烘干完的时间&…

Result of ‘BigDecimal.setScale()‘ is ignored

在Java中&#xff0c;BigDecimal.setScale() 方法用于格式化小数点后的位数&#xff0c;并可以选择舍入模式。如果你看到 "Result of BigDecimal.setScale() is ignored" 这样的警告&#xff0c;可能是因为你调用了 setScale() 方法&#xff0c;但没有将返回的新 Big…

Linux内核处理系统调用过程

Linux内核处理系统调用的过程是操作系统中一个非常关键的部分&#xff0c;它允许用户空间的程序请求内核提供的服务。以下是系统调用处理的一般步骤和概念&#xff1a; 1. **用户空间到内核空间的切换**&#xff1a; - 用户空间的程序通过执行系统调用指令&#xff08;如in…

arm、AArch64、x86、amd64、x86_64 的区别

arm vs AArch64 vs amd64 vs x86_64 vs x86 的区别 当涉及到 CPU 的时候&#xff0c;有许多术语&#xff1a;AArch64、x86_64、amd64、arm 等等。了解它们是什么以及它们之间的区别。 当你查看数据表或软件下载页面时是否被 ARM、AArch64、x86_64、i386 等术语混淆&#xff1f;…

运放构成电压跟随器,反馈电阻作用;运放电流采集电路,单电源供电,TINA仿真

电压跟随器 使用运放构成电压跟随器可以减小负载对信号源的影响&#xff0c;还可以提高信号带负载的能力&#xff0c;这是因为运放的结构特性&#xff0c;输入电阻大&#xff0c;输出电阻小。 是否决定使用该电压跟随器&#xff0c;就要看信号源&#xff0c;以及负载的阻抗大小…

【视觉SLAM】 十四讲ch5习题

1.*寻找一个相机&#xff08;你手机或笔记本的摄像头即可&#xff09;&#xff0c;标定它的内参。你可能会用到标定板&#xff0c;或者自己打印一张标定用的棋盘格。 参考我之前写过的这篇博客&#xff1a;【OpenCV】 相机标定 calibrateCamera Code来源是《学习OpenCV3》18.…

kaggle竞赛宝典 | 时序表示学习的综述!

本文来源公众号“kaggle竞赛宝典”&#xff0c;仅用于学术分享&#xff0c;侵权删&#xff0c;干货满满。 原文链接&#xff1a;时序表示学习的综述! 1 介绍 本文综述了时间序列数据中的通用表示学习方法&#xff0c;提出了一种新颖的分类方法&#xff0c;并讨论了其对提高…

闪电般的代码提示:在PyCharm中消灭延迟

闪电般的代码提示&#xff1a;在PyCharm中消灭延迟 PyCharm是一款功能强大的集成开发环境&#xff08;IDE&#xff09;&#xff0c;它提供了代码提示、自动完成等功能&#xff0c;极大地提高了开发效率。然而&#xff0c;在处理大型项目或复杂代码库时&#xff0c;代码提示可能…

Java 读取树状结构的.yaml文件

1. 编写yaml配置文件 以省市乡三级结构为例&#xff0c;配置文件如下&#xff1a; administrative:divisions:- code: BJname: 北京children:- code: BJ01name: 东城区children:- code: BJ0101name: 东华门街道- code: BJ0102name: 交道口街道- code: BJ0103name: 和平里街道…