【面试八股总结】MySQL事务:事务特性、事务并行、事务的隔离级别

参考资料:小林coding

一、事务的特性ACID

  • 原子性(Atomicity)

        一个事务是一个不可分割的工作单位,事务中的所有操作,要么全部完成,要么全部不完成,不会结束在中间某个环节。原子性是通过 undo log(回滚日志) 来保证的。

  • 一致性(Consistency)

        事务操作前和操作后,数据满足完整性约束,数据库保持一致性状态。

  • 隔离性(Isolation)

        数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致。多个事务并发执行时,每个事务都看不到其他事务的中间状态。每个事务都应该感觉就像它是唯⼀在数据库上运行的事务⼀样。隔离性是通过 MVCC(多版本并发控制)锁机制来保证的。

  • 持久性(Durability)

        ⼀旦事务被提交,其结果将永久保存在数据库中,即使系统发生故障。即使系统发生崩溃,事务的结果也不应该丢失,持久性是通过 redo log (重做日志)来保证的。

二、脏读、不可重复读、幻读

      并行事务是指多个事务同时执⾏,这可以提高数据库系统的性能和吞吐量。

1. 脏读:读到其他事务未提交的数据

        如果一个事务读到了另一个未提交事务修改过的数据,就意味着发生了「脏读」现象。如果另⼀个事务后来回滚,读取的数据就是无效的。

2. 不可重复读:前后读取的数据不⼀致

        在一个事务内多次读取同一个数据,如果出现前后两次读到的数据不一样的情况,就意味着发生了不可重复读现象。在事务执⾏期间其他事务可能修改了数据。

3. 幻读:前后读取的记录数量不⼀致

        在一个事务内多次查询某个符合查询条件的记录数量,如果出现前后两次查询到的记录数量不一样的情况,就意味着发生了幻读现象。在事务执⾏期间其他事务可能增加或删除了数据。

        脏读、不可重复读、幻读的现象会对事务的一致性产生不同程度的影响。严重性排序如下:

三、事务的隔离级别

        SQL 标准提出了四种隔离级别来规避脏读、不可重复读、幻读的现象,隔离级别越高,性能效率就越低,这四个隔离级别如下:

  • 读未提交(read uncommitted,指一个事务还没提交时,它做的变更就能被其他事务看到;
  • 读提交(read committed,指一个事务提交之后,它做的变更才能被其他事务看到;
  • 可重复读(repeatable read,指一个事务执行过程中看到的数据,一直跟这个事务启动时看到的数据是一致的,MySQL InnoDB 引擎的默认隔离级别
  • 串行化(serializable );会对记录加上读写锁,在多个事务对这条记录进行读写操作时,如果发生了读写冲突的时候,后访问的事务必须等前一个事务执行完成,才能继续执行;

        隔离水平高低排序如下:

         针对不同的隔离级别,并发事务时可能发生的现象也会不同。

  • 在「读未提交」隔离级别下,可能发生脏读、不可重复读和幻读现象;
  • 在「读提交」隔离级别下,可能发生不可重复读和幻读现象,但是不可能发生脏读现象;
  • 在「可重复读」隔离级别下,可能发生幻读现象,但是不可能脏读和不可重复读现象;
  • 在「串行化」隔离级别下,脏读、不可重复读和幻读现象都不可能会发生。

        MySQL InnoDB 引擎的默认隔离级别虽然是「可重复读」,但是它很⼤程度上可以避免幻读现象。解决的方案有两种:

  • 针对快照读(普通 select 语句),是通过 MVCC方式解决了幻读
  • 针对当前读:(select ... for update 等语句),是通过 next-key lock(记录锁+间隙锁)方式解决了幻读, 因为当执⾏ select ... for update 语句的时候,会加上 next-key lock,如果有其他事务在 next-key lock 锁范围内插入,那么这个插入语句就会被阻塞,无法成功插入,所以就很好了避免幻读问题。

隔离级别的实现

  • 读未提交

         可以读到未提交事务修改的数据,直接读取最新的数据;

  • 串行化

        通过加读写锁的方式来避免并行访问;

  • 读提交 和 可重复读

        通过 Read View 实现,区别在于创建 Read View 的时机不同,可以把 Read View 理解成一个数据快照。读提交是在每个语句执行前都会重新生成一个 Read View,而可重复读是启动事务时生成一个 Read View,然后整个事务期间都在用这个 Read View。

Read View 在 MVCC 里如何工作的?

Read View 有四个重要的字段:

  • m_ids :指的是在创建 Read View 时,当前数据库中「活跃事务」的事务 id 列表,注意是一个列表,“活跃事务”指的就是,启动了但还没提交的事务
  • min_trx_id :指的是在创建 Read View 时,当前数据库中「活跃事务」中事务 id 最小的事务,也就是 m_ids 的最小值。
  • max_trx_id :这个并不是 m_ids 的最大值,而是创建 Read View 时当前数据库中应该给下一个事务的 id 值,也就是全局事务中最大的事务 id 值 + 1;
  • creator_trx_id :指的是创建该 Read View 的事务的事务 id

        除此之外,聚簇索引记录中还包含两个隐藏列: 

  • trx_id,当一个事务对某条聚簇索引记录进行改动时,就会把该事务的事务 id 记录在 trx_id 隐藏列里
  • roll_pointer,每次对某条聚簇索引记录进行改动时,都会把旧版本的记录写入到 undo 日志中,然后这个隐藏列是个指针,指向每一个旧版本记录,于是就可以通过它找到修改前的记录。

在创建 Read View 后,我们可以将记录中的 trx_id 划分这三种情况:

⭐⭐⭐一个事务去访问记录的时候,除了自己的更新记录总是可见之外,还有这几种情况:

  • 如果记录的 trx_id 值小于 Read View 中的 min_trx_id 值,表示这个版本的记录是在创建 Read View 已经提交的事务生成的,所以该版本的记录对当前事务可见
  • 如果记录的 trx_id 值大于等于 Read View 中的 max_trx_id 值,表示这个版本的记录是在创建 Read View 才启动的事务生成的,所以该版本的记录对当前事务不可见
  • 如果记录的 trx_id 值在 Read View 的 min_trx_id 和 max_trx_id 之间,需要判断 trx_id 是否在 m_ids 列表中:
    • 如果记录的 trx_id  m_ids 列表中,表示生成该版本记录的活跃事务依然活跃着(还没提交事务),所以该版本的记录对当前事务不可见
    • 如果记录的 trx_id 不在 m_ids列表中,表示生成该版本记录的活跃事务已经被提交,所以该版本的记录对当前事务可见

这种通过「版本链」来控制并发事务访问同一个记录时的行为就叫 MVCC(多版本并发控制)。

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

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

相关文章

CSS-in-JS学习

CSS-in-JS CSS-in-JS 是一种将样式直接写入JavaScript代码中的方法,它通常与React、Vue等现代前端框架结合使用。 1. 什么是CSS-in-JS? CSS-in-JS 是一种编写样式的方法,它允许开发者在JavaScript组件内部定义样式,通常使用类似于CSS的语法。这种方式提高了代码的可复用…

C#根据数据量自动排版标签的样例

这是一个C#根据数据量自动排版标签的样例 using System; using System.Collections.Generic; using System.Data.SqlClient; using System.Drawing; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; using HslCommuni…

go mongo 唯一索引创建

1. 登录mongo,创建数据库 mongosh -u $username -p $password use test 2. 查看集合索引 db.$collection_name.getIndexes() 为不存在的集合创建字段唯一索引 package mainimport ("context""fmt""log""time""go…

代码随想录算法训练营第四十五天 | 1049. 最后一块石头的重量 II、494. 目标和、474.一和零

1049. 最后一块石头的重量 II 视频讲解: 动态规划之背包问题,这个背包最多能装多少?LeetCode:1049.最后一块石头的重量II_哔哩哔哩_bilibili 代码随想录 解题思路 直接将这一些石头,分为两堆,让他们尽可能…

假如Redis⾥面有1亿个key,其中有10w个key是以某个固定的已知的前缀开头的,如何将它们全部找出来?

使⽤用 keys 指令可以扫出指定模式的 key 列列表。但是要注意 keys 指令会导致线程阻塞⼀一段时间,线上服务会停 顿,直到指令执⾏行行完毕,服务才能恢复。这个时候可以使⽤用 scan 指令, scan 指令可以⽆无阻塞的提取出指定模式 的…

C语言 | Leetcode C语言题解之第120题三角形最小路径和

题目&#xff1a; 题解&#xff1a; int minimumTotal(int** triangle, int triangleSize, int* triangleColSize) {int f[triangleSize];memset(f, 0, sizeof(f));f[0] triangle[0][0];for (int i 1; i < triangleSize; i) {f[i] f[i - 1] triangle[i][i];for (int j …

SQL语句来实现不使用子查询的方式,直接通过JOIN和MAX函数来筛选出每个主表关联的最新子表记

除了使用JOIN和子查询的方式外&#xff0c;还可以使用窗口函数来实现不带子查询的方式来筛选出每个主表关联的最新子表记录。 以下是使用窗口函数的SQL语句示例&#xff1a; sql SELECT r.*, t.* FROM (SELECT r.*, t.*,ROW_NUMBER() OVER (PARTITION BY r.id ORDER BY t.creat…

latex中对目录的处理

文章目录 设置目录的章节编号宽度和章节标题的缩进设置条目的间距设置章节标题与页码之间的连接线 设置目录的章节编号宽度和章节标题的缩进 \usepackage{tocloft} \setlength{\cftsubsecnumwidth}{4cm} % 设置子章节编号的宽度为4cm \setlength{\cftsubsecindent}{1cm} % 设置…

【excel】设置二级联动菜单

文章目录 【需求】在一级菜单选定后&#xff0c;二级菜单联动显示一级菜单下的可选项【步骤】step1 制作辅助列1.列转行2.在辅助列中匹配班级成员 之前做完了 【excel】设置可变下拉菜单&#xff08;一级联动下拉菜单&#xff09;&#xff0c;开始做二级联动菜单。 【需求】在…

python实现——综合类型数据挖掘任务(无监督的分类任务)

综合类型数据挖掘任务 航空公司客户价值分析。航空公司客户价值分析。航空公司客户价值分析。航空公司已积累了大量的会员档案信息和其乘坐航班记录&#xff08;air_data.csv&#xff09;&#xff0c;以2014年3月31日为结束时间抽取两年内有乘机记录的所有客户的详细数据。利用…

万界星空科技MES系统功能介绍

制造执行系统或MES 是一个全面的动态软件系统&#xff0c;用于监视、跟踪、记录和控制从原材料到成品的制造过程。MES在企业资源规划(ERP) 和过程控制系统之间提供了一个功能层&#xff0c;为决策者提供了提高车间效率和优化生产所需的数据。 万界星空科技MES 系统基础功能&am…

Spark基础:Scala变量与数据类型

在Scala中&#xff0c;变量和数据类型是编程的基础。Scala作为一种强大的静态类型语言&#xff0c;支持多种数据类型&#xff0c;并提供了可变&#xff08;var&#xff09;和不可变&#xff08;val&#xff09;两种类型的变量声明方式。以下是在Scala中变量和数据类型的基础知识…

【全开源】Java短剧系统微信小程序+H5+微信公众号+APP 源码

打造属于你的精彩短视频平台 一、引言&#xff1a;为何选择短剧系统小程序&#xff1f; 在当今数字化时代&#xff0c;短视频已经成为人们日常生活中不可或缺的一部分。而短剧系统小程序源码&#xff0c;作为构建短视频平台的强大工具&#xff0c;为广大开发者提供了快速搭建…

03-树1 树的同构(浙大数据结构PTA习题)

03-树1 树的同构 分数 25 作者 陈越 单位 浙江大学 给定两棵树 T1​ 和 T2​。如果 T1​ 可以通过若干次左右孩子互换就变成 T2​&#xff0c;则我们称两棵树是“同构”的。例如图1给出的两棵树就是同构的&#xff0c;因为我们把其中一棵树的结点A、B、G…

CSPM.pdf

PDF转图片 归档&#xff1a;

.NET与C#和PLC交互的例子

要读取PLC中指定数据块&#xff08;DB&#xff09;内连续的多个字节&#xff0c;可以使用HSLCommunication库的Read方法。这个方法允许你一次性读取多个字节&#xff0c;然后再根据需要解析这些字节。 下面是修改后的代码&#xff0c;示范如何从指定数据块的起始地址读取30个字…

c与web前端:编程语言与前端技术的深度交融

c与web前端&#xff1a;编程语言与前端技术的深度交融 在编程领域&#xff0c;C语言和Web前端技术各自扮演着举足轻重的角色。C语言以其高效、灵活的特性&#xff0c;成为底层编程和系统级开发的利器&#xff1b;而Web前端技术则以其直观、交互的特点&#xff0c;为用户提供了…

跨境电商多店铺:怎么管理?风险如何规避?

跨境电商的市场辽阔&#xff0c;有非常多的商业机会。你可能已经在Amazon、eBay、Etsy等在线平台向潜在客户销售产品了。为了赚更多的钱&#xff0c;你可能还在经营多个店铺和品牌。 但是&#xff0c;像Amazon、eBay、Etsy等知名平台会有自己的规则&#xff0c;他们开发了很多…

ema_mnist_blog

使用ModelEmaV2优化MNIST分类模型 在深度学习模型的训练过程中&#xff0c;参数波动可能会导致模型在测试集上的性能不稳定。为了解决这个问题&#xff0c;可以使用指数移动平均&#xff08;EMA&#xff09;技术来平滑参数的更新&#xff0c;从而获得更稳定的模型。本文将介绍…

手拉手springboot整合kafka发送消息

环境介绍技术栈springbootmybatis-plusmysqlrocketmq软件版本mysql8IDEAIntelliJ IDEA 2022.2.1JDK17Spring Boot3.1.7kafka2.13-3.7.0 创建topic时&#xff0c;若不指定topic的分区(Partition主题分区数)数量使&#xff0c;则默认为1个分区(partition) springboot加入依赖kafk…