MySQL-事务&MVCC

前置文章:
一、MySQL-存储引擎

零、本文纲要

  • 一、事务
  • 二、MySQL事务原理
  • 三、redo log(重做日志)
  • 四、undo log(回滚日志)
  • 五、MVCC
    1、当前读
    2、快照读
    3、MVCC实现
    4、MVCC原理分析

tips:Ctrl + F快速定位所需内容阅读吧。

一、事务

1、事务介绍

事务是一组操作的集合,它是一个不可分割的工作单位,事务会把所有的操作作为一个整体一起向系统提交或撤销操作的请求,即这些操作要么同时成功,要么同时失败。

2、事务特性

  • 原子性(Atomicity):事务是不可分割的最小操作单元,要么全部成功,要么全部失败;
  • 一致性(Consistency):事务完成时,必须使所有的数据都保持一致状态;
  • 隔离性(Isolation):数据库系统提供的隔离机制,保证事务在不受外部并发操作影响的独立环境下运行;
  • 持久性(Durability):事务一旦提交或回滚,它对数据库中的数据的改变就是永久的。

二、MySQL事务原理

1、redo log & undo log

InnoDB中使用redo logundo log来保证事务的原子性、一致性、持久性。

2、锁 & MVCC

InnoDB中使用MVCC来保证事务的隔离性。

三、redo log(重做日志)

InnoDB架构.png

1、redo log介绍

redo log(重做日志),记录的是事务提交时数据页的物理修改,是用来实现事务的持久性

2、redo log作用

该日志文件由两部分组成:重做日志缓冲(redo log buffer)以及重做日志文件(redo log file),前者是在内存中,后者在磁盘中。
当事务提交之后会把所有修改信息都存到该日志文件中,用于在刷新脏页到磁盘,发生错误时,进行数据恢复使用。

3、对比redo log有无情形

对比redo log有无情形.png
  • 有redo log:

① 当对缓冲区的数据进行增删改之后,会首先将操作的数据页的变化,记录在redo log buffer中;
② 在事务提交时,会将redo log buffer中的数据刷新到redo log磁盘文件中;
③ 过一段时间之后,如果刷新缓冲区的脏页到磁盘时,发生错误,此时就可以借助于redo log进行数据恢复,这样就保证了事务的持久性;
④ 而如果脏页成功刷新到磁盘 或 或者涉及到的数据已经落盘,此时redolog就没有作用了,就可以删除了,所以存在的两个redolog文件是循环写的。

#ib_logfile0
#ib_logfile1

4、WAL

在业务操作中,我们操作数据一般都是随机读写磁盘的,而不是顺序读写磁盘。 而redo log在往磁盘文件中写入数据,由于是日志文件,所以都是顺序写的。顺序写的效率,要远大于随机写。这种先写日志的方式,称之为 WAL(Write-Ahead Logging)。

5、刷盘时机

innodb_flush_log_at_trx_commit:

0: 每秒将日志写入并刷新到磁盘一次;
1: 日志在每次事务提交时写入并刷新到磁盘,默认值;
2: 日志在每次事务提交后写入,并每秒刷新到磁盘一次。

四、undo log(回滚日志)

1、undo log介绍

回滚日志(也称撤销日志),用于记录数据被修改前的信息 , 作用包含两个 : 提供回滚(保证事务的原子性)MVCC(多版本并发控制)
undo log记录的是逻辑日志,可以认为记录的是与我们执行的update、insert、delete相反的逻辑语句。

2、undo log销毁

undo log在事务执行时产生,事务提交时,并不会立即删除undo log,因为这些日志可能还用于MVCC。

3、undo log存储

undo log采用段的方式进行管理和记录,存放在rollback segment回滚段中,内部包含1024个undo log segment。

五、MVCC

1、当前读

读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。

  • ① 对应SQL
  • select ... lock in share mode(共享锁);
  • select ...for update(排它锁);
  • update、insert、delete(排它锁)。
  • ② 演示示例

Ⅰ 事务B提交后,事务A当前读,可以读到最新数据,如下:

当前读.png

Ⅱ 事务B提交后,事务A快照读,不能读到最新数据,如下:

快照读.png

2、快照读

简单的select(不加锁)就是快照读,快照读,读取的是记录数据的可见版本,有可能是历史数据,不加锁,是非阻塞读。

  • ① 快照读的不同情形
  • Read Committed:每次select,都生成一个快照读;
  • Repeatable Read:开启事务后第一个select语句才是快照读的地方;
  • Serializable:快照读会退化为当前读。
  • ② 演示示例

具体示例见上。

3、MVCC实现

MVCC全称 Multi-Version Concurrency Control,多版本并发控制。指维护一个数据的多个版本,使得读写操作没有冲突,快照读为MySQL实现MVCC提供了一个非阻塞读功能。MVCC的具体实现,还需要依赖于数据库记录中的三个隐式字段undo log日志readView

  • ① 三个隐式字段

DB_TRX_ID:最近修改事务ID,记录插入这条记录或最后一次修改该记录的事务ID
DB_ROLL_PTR回滚指针,指向这条记录的上一个版本,用于配合undo log,指向上一个版本
DB_ROW_ID隐藏主键,如果表结构没有指定主键,将会生成该隐藏字段。

查看.ibd文件存储位置的命令:show global variables like '%datadir%';,位置为/var/lib/mysql/,如下:

 show global variables like '%datadir%';
+---------------+-----------------+
| Variable_name | Value           |
+---------------+-----------------+
| datadir       | /var/lib/mysql/ |
+---------------+-----------------+

查看表结构信息的指令,如下:

ibd2sdi ***.ibd

演示示例:

-- 创建未设置主键的tb_test02表格
create table tb_test02 (id int , name varchar(10));

-- 查看tb_test02表格的各列信息
[root@localhost test]# ibd2sdi tb_test02.ibd

-- 截取指令输出的关键内容
        "columns": [
            {
                "name": "id",
            },
            {
                "name": "name",
            },
            {
                "name": "DB_ROW_ID",
            },
            {
                "name": "DB_TRX_ID",
            },
            {
                "name": "DB_ROLL_PTR",
            }
        ],
  • ② undo log(回滚日志)

回滚日志(又称撤销日志),在insert、update、delete的时候产生的便于数据回滚的日志。

Ⅰ 当insert的时候,产生的undo log日志只在回滚时需要,在事务提交后,可被立即删除
Ⅱ 而updatedelete的时候,产生的undo log日志不仅在回滚时需要,在快照读时也需要,不会立即被删除

注意:insert对快照读已存在的数据不产生影响,而update和delete可能产生影响,所以此处会有区别。

版本链:不同事务或相同事务对同一条记录进行修改,会导致该记录的undolog生成一条记录版本链表,链表的头部最新的旧记录,链表尾部最早的旧记录。

image.png
  • ③ readview(读视图)

ReadView(读视图)是 快照读 SQL执行时MVCC提取数据的依据,记录并维护系统当前活跃的事务(未提交的)id

四个核心字段:
m_ids:当前活跃的事务ID集合;
min_trx_id:最小活跃事务ID;
max_trx_id预分配事务ID,当前最大事务ID+1(因为事务ID是自增的);
creator_trx_id:ReadView创建者的事务ID。

版本链数据的访问规则(trx_id是当前undolog版本链对应事务ID):

访问规则.png

生成ReadView的时机:
Ⅰ READ COMMITTED :在事务中每一次执行快照读时生成ReadView;
Ⅱ REPEATABLE READ:仅在事务中第一次执行快照读时生成ReadView,后续复用该ReadView。

4、MVCC原理分析

  • ① RC(READ COMMITTED)隔离级别

RC隔离级别下,在事务中每一次执行快照读时生成ReadView。

每一次执行快照读时生成ReadView.png

示例解读:

演示示例RC.png

此情形中,当我们对比到版本链中trx_id = 2的时候,满足条件②,此时该版本就是版本链中事务5可以读取到的数据。

  • ② RR(REPEATABLE READ)隔离级别

RR隔离级别下,仅在事务中第一次执行快照读时生成ReadView,后续复用该ReadView。RR 是可重复读,在一个事务中,执行两次相同的select语句,查询到的结果是一样的。

第一次执行快照读时生成ReadView.png

六、结尾

以上即为事务和MVCC部分的内容了,感谢阅读。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容