我的编程空间,编程开发者的网络收藏夹
学习永远不晚

如何浅析mysql事务和隔离级别底层原理

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

如何浅析mysql事务和隔离级别底层原理

本篇文章给大家分享的是有关如何浅析mysql事务和隔离级别底层原理,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

    前言

    首先回顾一下什么是事务,事务是数据库操作的最小工作单元,是作为单个逻辑工作单元执行的一系列操作;这些操作作为一个整体一起向系统提交,要么都执行、要么都不执行;事务是一组不可再分割的操作集合(工作逻辑单元)。

    事务的特性:

    原子性(Atomicity):原子性是指事务包含的所有操作要么全部成功,要么全部失败回滚。一致性(Consistency):事务执行的结果必须是使数据库从一个一致性状态变到另一个一致性状态。隔离性(Isolation):一个事务的执行不能其它事务干扰。即一个事务内部的操作及使用的数据对其它并发事务是隔离的,并发执行的各个事务之间不能互相干扰。持久性(Durability):指一个事务一旦提交,它对数据库中的数据的改变就应该是永久性的。接下来的其它操作或故障不应该对其执行结果有任何影响。

    一、事务底层原理浅析

    原子性:

    实现原理:undo log

    undo log也被成为回滚日志,它是事务实现原子性和隔离性的基础。当事务对数据库进行修改时,InnoDB会生成对应的undo log;如果事务执行失败或调用了rollback,导致事务需要回滚,便可以利用undo log中的信息将数据回滚到修改之前的样子。

    undo log属于逻辑日志,它记录的是sql执行相关的信息。当发生回滚时,InnoDB会根据undo log的内容做与之前相反的工作:对于每个insert,回滚时会执行delete;对于每个delete,回滚时会执行insert;对于每个update,回滚时会执行一个相反的update,把数据改回去。

    undo 存放在数据库内部的一个特殊段segment中,这个段称为undo段。undo段位于共享表空间中。undo是逻辑日志,因此只是将数据库逻辑的恢复到原来的样子。undo log会产生redo log,也就是undo log的产生会伴随着redo log的产生,因为undo log也需要持久性的保护。

    undo log执行记录是在每次写入数据或者修改数据之前。

    如何浅析mysql事务和隔离级别底层原理

    undo log使用原理:数据库表每行数据会多两列DATA_TRX_ID和DATA_ROLL_PTR(可能还有一列DB_ROW_ID,当没有默认主键时会自动加上这列)。DATA_TRX_ID表示当前数据的事务版本, DATA_ROLL_PTR 则指向刚刚拷贝到

    undo log 链中的旧版本记录,undo log是个链表,如果多个事务多次修改会继续生成undo log并通过DATA_ROLL_PTR建立指向关系。用图说明一下:

    如何浅析mysql事务和隔离级别底层原理

    这样,一旦事务发生回滚,mysql便可以借助undo log实现数据还原,从而保证未提交事务的原子性。

    持久性

    实现原理:redo log

    由于InnoDB作为MySQL的存储引擎,数据是存放在磁盘中的,为了减少磁盘IO,提高读取性能InnoDB提供了缓存池——Buffer Pool。Buffer Pool中包含了磁盘中部分数据页的映射,作为访问数据库的缓冲:当从数据库读取数据时,会首先从Buffer Pool中读取,如果Buffer Pool中没有,则从磁盘读取后放入Buffer Pool;当向数据库写入数据时,会首先写入Buffer Pool,Buffer Pool中修改的数据会定期刷新到磁盘中(这一过程称为刷脏)。

    不过这也带来了一个新的问题,如果MySQL宕机,而此时Buffer Pool中修改的数据还没有刷新到磁盘,就会导致数据的丢失,事务的持久性无法保证。

    为了解决这个问题就引入了redo log,也叫重做日志。当数据修改时,除了修改Buffer Pool中的数据,还会在redo log记录这次操作;当事务提交时,会调用fsync接口对redo log进行刷盘。如果MySQL宕机,重启时可以读取redo log中的数据,对数据库进行恢复。redo log采用的是WAL(Write-ahead logging,预写式日志),所有修改在提交前先写入日志,保证了数据不会因MySQL宕机而丢失,从而满足了持久性要求。redo log是针对物理页的,并发执行,最后一次提交会覆盖未提交的数据。本地redo log:

    如何浅析mysql事务和隔离级别底层原理

    redo log也是有缓冲区的——redo log buffer,当事务提交之后会把所有修改信息都刷新到磁盘上。用户也可以通过控制通过变量 innodb_flush_log_at_trx_commit 的值来修改刷新策略(默认1),如设置值为2,控制成每秒刷新,这样事务提

    交就会较快,不过可能面临日志丢失的风险。

    redo log是在SQL语句执行之后记录的。

    如何浅析mysql事务和隔离级别底层原理

    既然redo log也需要存储,也涉及磁盘IO为啥还用它?

    (1)redo log 的存储是顺序存储,而缓存同步是随机操作。

    (2)缓存同步是以数据页为单位的,每次传输的数据大小大于redo log。

    redo log是用来恢复数据的 用于保障已提交事务的持久化特性。

    隔离性:

    原理:
    (1). 写操作对写操作的影响:锁机制保证隔离性
    (2). 写操作对读操作的影响:MVCC保证隔离性

    一致性:

    一致性比较特殊,前面的原子性、持久性和隔离性都是为一致性服务的,除此之外,一致性还依赖于数据库自我提供的保障,如SQL语法验证,列类型插入数据类型验证,同时也依赖于应用层的保障,如转账操作,需要开发人员进行转账者的余额扣除和接受者的余额增加,如果应用层面出现问题,那么一致性也是无法保障的。

    二、隔离级别底层原理浅析

    在事务底层原理浅析中,关于隔离性的原理没有过多深入,在此我们简单介绍一下。

    首先介绍一下mysql的MVCC(MultiVersion Concurrency Control) 叫做多版本并发控制。它是依赖undo logread view实现的。undo log上文已经介绍过了,不再赘述,read view(可读视图),这与我们平时理解的数据库视图不同,它是用来判断当前数据版本的可见性的。

    readview主要有四个属性:

    (1). m_ids 代表生成ReadView时,当前所有活跃的事务ID,活跃的意思就是事务开启了还没提交;

    (2). min_trx_id 表示当前活跃的mIds中最小的事务ID;

    (3). max_trx_id 表示生成ReadView时,最大的事务ID,它不一定是mIds中最大的事务ID;

    (4).creator_trx_id表示创建该ReadView的事务ID。

    如何浅析mysql事务和隔离级别底层原理

    注意:每开启一个事务,事务ID就自增一次,事务ID可以看做一个全局自增变量。最先开启的事务不一定比后开启的事务先提交,比如长连接,所以不要认为max_trx_id就是mIds中的最大值。

    read view是怎样借助上面四个属性,判断事务应该读取那个版本的数据呢?

    如果被访问版本的 data_trx_id 小于 m_ids 中的最小值,说明生成该版本的事务在 ReadView 生成前就已经提交了,那么该版本可以被当前事务访问。

    如果被访问版本的 data_trx_id 属性值与ReadView中的creator_trx_id值相同,意味着当前事务在访问它自己修改过的记录,所以该版本可以被当前事务访问。

    如果被访问版本的 data_trx_id 属性值大于ReadView中的max_trx_id值,表明生成该版本的事务在当前事务生成ReadView后才开启,所以该版本不可以被当前事务访问。

    如果被访问版本的 data_trx_id 属性值在ReadView的min_trx_id和max_trx_id之间,那就需要判断一下trx_id属性值是不是在m_ids列表中,如果在,说明创建ReadView时生成该版本的事务还是活跃的,该版本不可以被访问;如果不在,说明创建ReadView时生成该版本的事务已经被提交,该版本可以被访问。

    当一个事务要读取一行数据,首先用上面规则判断数据的最新版本也就是那行记录,如果发现可以访问就直接读取了,如果发现不能访问,就通过DATA_ROLL_PTR指针找到undo log,递归往下去找每个版本,知道读取到自己可以读取的版本为止,如果读取不到就返回空。

    所以访问数据时,数据库里面会创建一个视图,访问的时候以视图的逻辑结果为准:

    READ UNCOMMITED (未提交读):此隔离级别下直接返回记录上的最新值,没有视图概念。因为读不会加任何锁,所以写操作在读的过程中修改数据,所以会造成脏读。好处是可以提升并发处理性能,能做到读写并行。

    READ COMMITED (提交读):此隔离级别下,这个视图是在每个 SQL语句开始执行的时候创建的。InnoDB在 READ COMMITTED,使用排它锁,读取数据不加锁而是使用了MVCC机制。或者换句话说他采用了读写分离机制。

    REPEATABLE READ (可重复读):此隔离级别下,这个视图是在事务启动时创建的,整个事务存在期间都用这个视图。

    SERIALIZABLE (串行化):此隔离级别下直接用加锁的方式来避免并行访问。

    写到这里,你也许发现了,MySQL中借助MVCC在可重复读隔离级别下其实也杜绝了幻读的发生。

    三、总结

    原子性:使用 undo log (回滚日志)实现回滚,从而保障未提交事务的原子性;

    持久性:使用 redo log(重做日志)实现数据恢复,从而保障已提交事务的持久性;

    隔离性:使用锁以及MVCC思想实现读写分离,读读并行,读写并行;

    一致性:通过回滚、恢复和在并发环境下的隔离做到一致性。

    以上就是如何浅析mysql事务和隔离级别底层原理,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注编程网行业资讯频道。

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    如何浅析mysql事务和隔离级别底层原理

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档

    猜你喜欢

    如何浅析mysql事务和隔离级别底层原理

    本篇文章给大家分享的是有关如何浅析mysql事务和隔离级别底层原理,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。前言首先回顾一下什么是事务,事务是数据库操作的最小工作单元,是作
    2023-06-21

    Mysql事务隔离级别原理实例解析

    引言 大家在面试中一定碰到过 说说事务的隔离级别吧? 老实说,事务隔离级别这个问题,无论是校招还是社招,面试官都爱问!然而目前网上很多文章,说句实在话啊,我看了后我都怀疑作者弄懂没!因为他们对可重复读(Repeatable Read)和串行
    2022-05-29

    浅析MySQL如何实现事务隔离

    目录一、前言二、RC与RR隔离级别2.1、RR事务隔离级别下查询结果2.2、RC事务隔离级别下查询结果三、事务隔离在MVCC的实现3.1、数据行ROW的多版本3.2、视图数组3.3、一致性视图3.4、当前读与快照读3.4.1、当前读与快照读
    2022-05-16

    如何实现MySQL底层优化:事务的并发控制和隔离级别选择

    如何实现MySQL底层优化:事务的并发控制和隔离级别选择摘要:在MySQL数据库中,事务的并发控制和隔离级别的选择对于数据库性能和数据一致性非常重要。本文将介绍如何通过底层优化来实现MySQL事务的并发控制和隔离级别选择,并提供具体的代码示
    如何实现MySQL底层优化:事务的并发控制和隔离级别选择
    2023-11-08

    【MySQL】搞懂ACID原则和事务隔离级别

    宜未雨而绸缪,毋临渴而掘井说说MySQL的事务数据库事务(Database Transaction) ,是指作为单个逻辑工作单元执行的一系列操作,要么完全地执行,要么完全地不执行。一个数据库事务通常包含对数据库进行读或写的一个操作序列。它的存在包含有以下两个目
    【MySQL】搞懂ACID原则和事务隔离级别
    2015-04-28

    mysql如何查看事务隔离级别

    要查看MySQL数据库的当前事务隔离级别,可以执行以下命令:SELECT @@tx_isolation;该命令将返回当前数据库的事务隔离级别,可能的返回值包括:READ UNCOMMITTEDREAD COMMITTEDREPEATA
    mysql如何查看事务隔离级别
    2024-04-17

    MySQL事务与隔离级别如何使用

    这篇文章主要介绍“MySQL事务与隔离级别如何使用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“MySQL事务与隔离级别如何使用”文章能帮助大家解决问题。什么是事务事务就是一组操作的集合,事务将整组
    2023-07-05

    MySQL事务及Spring隔离级别实现原理详解

    1、事务具有ACID特性原子性(atomicity):一个事务被事务不可分割的最小工作单元,要么全部提交,要么全部失败回滚。一致性(consistency):数据库总是从一致性状态到另一个一致性状态,它只包含成功事务提交的结果隔离型(iso
    2022-05-18

    详解MySQL中事务隔离级别的实现原理

    前言说到数据库事务,大家脑子里一定很容易蹦出一堆事务的相关知识,如事务的ACID特性,隔离级别,解决的问题(脏读,不可重复读,幻读)等等,但是可能很少有人真正的清楚事务的这些特性又是怎么实现的,为什么要有四个隔离级别。 今天我们就先来聊聊M
    2022-05-20

    MySQL是如何实现事务的隔离级别

    摘要本文旨在了解MySQL InnoDB引擎如何支持事务的隔离级别。文章主要内容分两个部分。第一部分阐述数据库的并发问题以及为之产生的ANSI SQL 标准隔离级别。第二部分根据 MySQL 官方文档解释 InnoDB 是如何支持这些隔离级别的。数据库事务的并
    MySQL是如何实现事务的隔离级别
    2020-07-01

    如何查看当前MySQL的事务隔离级别?

    通过执行SELECT @@TX_ISOLATION命令我们可以检查当前MySQL事务隔离级别。示例mysql> SELECT @@TX_ISOLATION;+-----------------+| @@TX_ISOLATION |+
    2023-10-22

    编程热搜

    目录