MySQL事务

目录

 一.两个例子

二.什么是事务

三.为什么出现事务

四.事务的版本支持

五.事务提交方式

 六.事务常见操作方式

1.隔离性更改

2.表创建与自动提交

3.事务开始与回滚

4.事务提交

5.关闭自动提交

6.结论

七.事务隔离级别

1.理解隔离性

2.隔离级别

(1)查看与设置隔离性

(2)读未提交【Read Uncommitted】

(3) 读提交【Read Committed】

(4)可重复读【Repeatable Read】

(5)串行化【serializable】

(6)总结

八.一致性

九.再次理解隔离性

1.数据库三种并发的场景

2.读-写

(1)MVCC

(2)3个隐藏列字段

(3)undo日志

(4)模拟MVCC 

(5)Read View

十.RR与RC的本质区别


前言:MySQL的最后一大重点,也是一大难点,事务。这一篇会先去理解一下什么是事务,再从事务的四个特性原子性、一致性、隔离性、持久性进行具体介绍。

一.两个例子

(1)这是在一个银行中,张三向李四转钱的例子:

 (2)火车票售票系统例子:

        想要解决这个问题,所要满足的属性:

① 买票的过程得是原子的
② 买票互相不能影响
③ 买完票要永久有效
④ 买前和买后都要是确定的状态

二.什么是事务

        事务就是一组DML语句组成,这些语句在逻辑上存在相关性,这一组DML语句要么全部成功,要么全部失败,是一个整体。MySQL提供一种机制,保证我们达到这样的效果。事务还规定不同的客户端看到的数据是不相同的(一个查,一个改,查的那个每次查到的数据可能都是不同的)。

        事务就是要做的或所做的事情,主要用于处理操作量大,复杂度高的数据。假设一种场景:你毕业了,学校的教务系统后台 MySQL 中,不在需要你的数据,要删除你的所有信息(一般不会) , 那么要删除你的基本信息(姓名,电话,籍贯等)的同时,也删除和你有关的其他信息,比如:你的各科成绩,你在校表现,甚至你在论坛发过的文章等。这样,就需要多条 MySQL 语句构成,那么所有这些操作合起来,就构成了一个事务。

        正如我们上面所说,一个 MySQL 数据库,可不止你一个事务在运行,同一时刻,甚至有大量的请求被包装成事务,在向 MySQL 服务器发起事务处理请求。而每条事务至少一条 SQL ,会有很多 SQL ,这样如果大家都访问同样的表数据,在不加保护的情况,就绝对会出现问题。甚至,因为事务由多条 SQL 构成,那么,也会存在执行到一半出错或者不想再执行的情况,那么已经执行的怎么办呢?

        所以,一个完整的事务,绝对不是简单的 sql 集合,还需要满足如下四个属性:

        ① 原子性:一个事务(transaction)中的所有操作,要么全部完成,要么全部不完成,不会结束在中间某个环节。事务在执行过程中发生错误,会被回滚(Rollback)到事务开始前的状态,就像这个事务从来没有执行过一样。

        ② 一致性:在事务开始之前和事务结束以后,数据库的完整性没有被破坏。这表示写入的资料必须完全符合所有的预设规则,这包含资料的精确度、串联性以及后续数据库可以自发性地完成预定的工作

        ③ 隔离性:数据库允许多个并发事务同时对其数据进行读写和修改的能力,隔离性可以防止多个事务并发执行时由于交叉执行而导致数据的不一致。事务隔离分为不同级别,包括读未提交( Readuncommitted )、读提交( read committed )、可重复读( repeatable read )和串行化( Serializable )

        ④ 持久性:事务处理结束后,对数据的修改就是永久的,即便系统故障也不会丢失。

        上面四个属性,可以简称为 ACID,AID共同保证C。

原子性(Atomicity,或称不可分割性)
一致性(Consistency)
隔离性(Isolation,又称独立性)
持久性(Durability)

        一致性仅仅是概念上的,当原子性、隔离性、持久性这三点做到了,那么结果就是一致性的。

        因此,原子性、隔离性、持久性是手段,而一致性是目的。

        用原子性、隔离性、持久性这三种属性封装成事务,通过使用这个事务来达到一致性的目的。一致性是上层的概念,要站在使用者的角度(以上面的银行例子为例:转账前和转账后它的结果是可预期的,数据是一致的,不会出现差错,出现差错也会自动回滚)

        mysqld要提供事务机制,就注定了mysqld内部要提供编码和数据结构的支持。

        mysqld一定会同时存在多个事务,就注定了mysqld要对多个事务进行管理工作:先描述,再组织。

        事务不要抽象的理解它,事务最终一定是要以某种 数据结构+算法 管理起来的。

三.为什么出现事务

        事务被 MySQL 编写者设计出来,本质是为了当应用程序访问数据库的时候,事务能够简化我们的编程模型,不需要我们去考虑各种各样的潜在错误和并发问题.可以想一下当我们使用事务时,要么提交,要么回滚,我们不会去考虑网络异常了,服务器宕机了,同时更改一个数据怎么办。因此事务本质上是为了应用层服务的.而不是伴随着数据库系统天生就有的.

        后面把 MySQL 中的一行信息,称为一行记录。

四.事务的版本支持

        在 MySQL 中只有使用了 Innodb 数据库引擎的数据库或表才支持事务, MyISAM 不支持。

        查看数据库引擎:

show engines;

        InnoDB的Comment中显示,Supports transactions,就说明他支持事务。

五.事务提交方式

        事务的提交方式常见的有两种:

① 自动提交

② 手动提交

        查看事务提交方式:

        这里表明事务默认是会自动提交的。

       用 SET 来改变 MySQL 的自动提交模式:

        set为0就是关闭自动提交。

        set为1就是开启自动提交。

 六.事务常见操作方式

1.隔离性更改

select @@tx_isolation; //查看隔离性

        这个隔离级别默认是repeatable read,我们为了便于演示,将其默认隔离级别设置成读未提交:read uncommited。

        但是这里并没有改变,是因为我们要重启mysql才可以变,这就与隔离性有关了。

       

        重启后,就变了。 

2.表创建与自动提交

        创建表:

show variables like 'autocommit'; // 查看是否是否自动提交

set autocommit=1; // 开启自动提交

set autocommit=0; // 关闭自动提交 (手动提交)

        这里查看事务是否自动提交,默认是自动提交的。

3.事务开始与回滚

start transaction; // 开始事务

begin; // 开始事务

        这两种写法都可以。

savepoint 保存点名; // 创建一个保存点

rollback; 回滚到最开始

rollback to 保存点名; // 回滚到这个保存点处

        这里创建一个事务,设立保存点1,并插入一个数据。

        这里因为我们设置了读未提交,所以即使没有提交也是可以读到的。

        再插入一个,也是可以看到的。

        rollback回滚到最开始。

        因为回滚到最开始,所以数据和保存点全部被丢弃了。

        这里重新插入一下。

        这里回滚到s2。

 

        可以看到数据也变成了s2时的只有一个数据。

        再回滚到s1,就没有数据了。

4.事务提交

commit; // 提交事务

        重新开启事务,插入数据。

        可以看到数据。

        这里我们不commit,而是ctrl+\,异常终止掉。

 

        会发现数据都没了。

        当然,mysql崩溃或者客户端崩溃都是一样的,都会终止。

        这里体现了mysql事务的原子性。

        这里正常commit。 

        mysql异常终止。

        这时依旧有数据,因为我们正常commit了。

        这里体现了mysql事务的持久性。

5.关闭自动提交

        之前一直都是在自动提交的情况下测试的,接下来关闭自动提交,即采取手动提交的方式来测试。

        关闭了自动提交。

        清空表之后,开启事务,然后异常终止。

        会发现依旧没有数据。

        如果commit了,那么也和之前一样会有数据。

那么这个自动提交难道不是像他的名字一样,可以自动提交吗?

        当然是可以自动提交,但是使用begin和start transaction是手动开启的事务,必须要手动结束,手动开启的事务是不受是否自动提交影响的。

        这个自动提交影响的是不手动开启事务时的单个sql语句(单个事务),接下来演示:

        我们先开启自动提交。

        插入两个数据,然后退出。

        然后可以发现数据正常保存了,以前就是这样的。 

        删除掉表内记录。

        当我们关闭自动提交。

 

         再插入两个数据,退出。

        发现这个数据没有了。

        这时因为单sql默认都是会以事务的方式进行提交的,只不过这个事务只有一个sql语句。

        如果没有用begin和start transaction开启事务,我们就默认执行我们的单sql,就是一个事务,sql执行完毕后会按照autocommit来决定是否自动提交。

        这些sql语句也不只是上面测试所用的插入语句,删除、修改等sql语句如果关闭了自动提交,那么退出sql后,就会回滚到未使用改sql语句之前。如果关闭了自动提交,又想要单个sql语句生效,就需要手动commit。

6.结论

① 只要输入begin或者start transaction,事务只有通过commit提交,才会持久化,与是否设置set autocommit无关。

② 事务可以手动回滚,同时,当操作异常,MySQL会自动回滚
③ 对于 InnoDB 每一条 SQL 语言都默认封装成事务,自动提交。(select有特殊情况,因为MySQL 有 MVCC )

④ 从上面的例子,我们能看到事务本身的原子性(回滚),持久性(commit)

       

        注意事项:

① 如果没有设置保存点,也可以回滚,只能回滚到事务的开始。直接使用 rollback(前提是事务还没有提交)
② 如果一个事务被提交了(commit),则不可以回滚(rollback)
③ 可以选择回退到哪个保存点
④ InnoDB 支持事务, MyISAM 不支持事务
⑤ 开始事务可以使 start transaction 或者 begin

七.事务隔离级别

1.理解隔离性

        MySQL服务可能会同时被多个客户端进程(线程)访问,访问的方式以事务方式进行

        一个事务可能由多条SQL构成,也就意味着,任何一个事务,都有执行前,执行中,执行后的阶段。而所谓的原子性,其实就是让用户层,要么看到执行前,要么看到执行后。执行中出现问题,可以随时回滚。所以单个事务,对用户表现出来的特性,就是原子性。

        但,毕竟所有事务都要有个执行过程,那么在多个事务各自执行多个SQL的时候,就还是有可能会出现互相影响的情况。比如:多个事务同时访问同一张表,甚至同一行数据。

        就如同你的父母跟你说:你要么别学,要学就学到最好。至于你怎么学,中间有什么困难,你的父母不关心。那么你的学习,对你的父母来讲,就是原子的。那么你学习过程中,很容易受别人干扰,此时,就需要将你的学习隔离开,保证你的学习环境是健康的。

        数据库中,为了保证事务执行过程中尽量不受干扰,就有了一个重要特征:隔离性

        数据库中,允许事务受不同程度的干扰,就有了一种重要特征:隔离级别

2.隔离级别

        隔离级别是在隔离事务这件事情上,做的不同程度的妥协,前提依旧要保证数据安全。

        读未提交【Read Uncommitted】: 在该隔离级别,所有的事务都可以看到其他事务没有提交的执行结果。(实际生产中不可能使用这种隔离级别的),但是相当于没有任何隔离性,也会有很多并发问题,如脏读,幻读,不可重复读等,我们上面为了做实验方便,用的就是这个隔离性。

        读提交【Read Committed】 :该隔离级别是大多数数据库的默认的隔离级别(不是 MySQL 默认的)。它满足了隔离的简单定义:一个事务只能看到其他的已经提交的事务所做的改变。这种隔离级别会引起不可重复读,即一个事务执行时,如果多次 select, 可能得到不同的结果。

        可重复读【Repeatable Read】: 这是 MySQL 默认的隔离级别,它确保同一个事务,在执行中,多次读取操作数据时,会看到同样的数据行。但是会有幻读问题。

        串行化【Serializable】: 这是事务的最高隔离级别,它通过强制事务排序,使之不可能相互冲突,从而解决了幻读的问题。它在每个读的数据行上面加上共享锁,但是可能会导致超时和锁竞争(这种隔离级别太极端,实际生产基本不使用)

        隔离级别如何实现:隔离,基本都是通过锁实现的,不同的隔离级别,锁的使用是不同的。常见有,表锁,行锁,读锁,写锁,间隙锁(GAP),Next-Key锁(GAP+行锁)等。不过,我们目前现有这个认识就行,先关注上层使用。

        事务中,所谓的提交commit,并不是把数据进行刷盘,这个刷盘的过程是mysqld自己会执行的。

        commit设置事务的状态,表示该数据已经算是交付给了mysqld。

(1)查看与设置隔离性

SELECT @@global.tx_isolation; --查看全局隔离级别

SELECT @@session.tx_isolation; --查看会话(当前)隔离级别

SELECT @@tx_isolation; --查看会话(当前)隔离级别

        会话的隔离级别:本次登陆对应的隔离级别(重新登陆刷新为全局隔离级别)

        全局隔离级别:本次mysql对应的隔离级别(重启mysqld刷新为默认隔离级别)

        这里设置会话隔离级别为read committed。

        可以发现全局隔离级别并没有变。

        退出,重新进mysql。

        会发现全部变回去了。因为会话退出了,重新登陆了,就要用全局的隔离级别。

        重启mysqld。

        这时可以发现既没有用会话的隔离级别,也没有用全局的隔离级别。这里用的的mysql的默认隔离级别repeatable read。 

(2)读未提交【Read Uncommitted】

        设置为读未提交。 

        这里事务内部的sql一执行,另一个连接该sql的客户端立刻就可以看到,这就说明它没有任何隔离性。如果两个客户端在运行两个事务,能看到对方没提交的事务记录,就可能会互相影响,出现交替的情况,引起并发问题。

        一个事务在执行中,读到另一个执行中事务的更新(或其他操作)但是未commit的数据,这种现象叫做脏读,读到的数据叫脏数据。

        几乎没有加锁,虽然效率高,但是问题太多,不建议采用。

(3) 读提交【Read Committed】

        更改全局隔离级别为read committed。 

         可以发现只有全局的变了,那么我们退出重进一下就好了。

        都变成read committed隔离级别了。

        这里是自动提交的,删除id为2的记录。 

        另一个客户端显示的信息中id为2的记录被删除了。这个是应该的, 因为单sql语句就是事务,执行完自动提交了,所以另一客户端的mysql会看到新的查询信息。

        这里开启一个事务,删除id为1的记录,会发现本客户端查询时因为都被删掉了所以没有记录了,而另一客户端查询时,依旧可以查询到id为1的记录,因为未提交,所以其它客户端不会执行该客户端处于事务内的sql语句。 

        只有commit了,另一客户端才能把该客户端事务内的sql语句执行。 

        再插入两个数据。

        两个客户端都开启一个事务,右边先看一下emp表只有张三和李四。

        左边更新张三为小六,select发现更新成功,与此同时右边select发现没有改变。

        左边插入十全。select发现插入成功,与此同时右边select发现没有改变。

        左边commit,右边select,发现更新sql语句和插入sql语句都执行了。

        这种现象其中一端commit,另一端才能看到的现象就是读提交。但这是存在问题的,这一端提交了,也不应该被正在和你一起运行的另一端事务看到。

        此时还在当前事务中,并未commit,那么就造成了,同一个事务内,同样的读取,在不同的时间段(依旧还在事务操作中),读取到了不同的值,这种现象叫做不可重复读(non reapeatable read)。

        这也是一个问题,正常应该让两个事务各做各的,互不影响。假设某个商场根据购买者的金额送相应的礼品,这就需要两个事务,一个事务用来记录这个购买者的花费金额,另一个事务用来查询购买者处于哪一个购买挡位。假设购买者现在花了1000元,在1000-2000元这个档,送一个手表;然后购买者这个事务提交了,查询事务因为一些原因暂停了,但是此时查询到了这个购买者处于1000-2000元这个档,这时购买者又购买了一些东西,达到了2000-3000元这个档,那么等到查询事务不再暂停时,继续往下执行就会查询到购买者处于2000-3000这个档,就又会送一个首饰。那这是有问题的,当查询事务暂停的时候,另一个购买者记录事务修改后commit就不应该影响到这个查询事务。

(4)可重复读【Repeatable Read】

        修改隔离级别为可重复读。

        这里可以看到,删除和插入时,这一端都是可以看到的,而另一端是无法看到的,这一点和上面的读提交是一样的。

        但是这一端commit后,另一端select依旧没有变,直到另一端commit了,才能看到。

        这就是可重复读。

        可重复读本质其实就是一个事务内部,不受任何事务操作的影响,每次查到的数据都是一致的。

         MySQL默认采用的隔离级别就是可重复读。

        多次查看,发现终端A在对应事务中insert的数据,在终端B的事务周期中,也没有什么影响。也符合可重复的特点。

        但是,一般的数据库在可重复读情况的时候,无法屏蔽其他事务insert的数据(为什么?因为隔离性实现是对数据加锁完成的,而insert待插入的数据因为并不存在,那么一般加锁无法屏蔽这类问题),会造成虽然大部分内容是可重复读的,但是insert的数据在可重复读情况被读取出来,导致多次查找时,会多查找出来新的记录,就如同产生了幻觉。这种现象,叫做幻读(phantom read)。

        而MySQL在RR级别的时候,是解决了幻读问题的(解决的方式是用Next-Key锁(GAP+行锁)解决的)。

(5)串行化【serializable】

        将隔离级别改为串行化。

        这里两端事务开启的时候,可以进行查询操作,但是其中一端想要修改,这就会被阻塞住,因为是串行化的,存在另一端开启了事务,因此这一端无法进行插入、删除、修改的操作。

        只有另一端commit了,这一端才会继续执行。

        这里如果长时间另一端不commit,就会报错。

(6)总结

        ① 其中隔离级别越严格,安全性越高,但数据库的并发性能也就越低,往往需要在两者之间找一个平衡点。

        ② 不可重复读的重点是修改和删除:同样的条件, 你读取过的数据,再次读取出来发现值不一样了幻读的重点在于新增:同样的条件, 第1次和第2次读出来的记录数不一样

        ③ 说明: mysql 默认的隔离级别是可重复读,一般情况下不要修改
        ④ 上面的例子可以看出,事务也有长短事务这样的概念。事务间互相影响,指的是事务在并行执行的时候,即都没有commit的时候,影响会比较大。

        MySQL不同隔离级别的不同问题(注意这里是MySQL,不是普通的数据库): 

八.一致性

        事务执行的结果,必须使数据库从一个一致性状态,变到另一个一致性状态。当数据库只包含事务成功提交的结果时,数据库处于一致性状态。如果系统运行发生中断,某个事务尚未完成而被迫中断,而改未完成的事务对数据库所做的修改已被写入数据库,此时数据库就处于一种不正确(不一致)的状态。因此一致性是通过原子性来保证的。

        其实一致性和用户的业务逻辑强相关,一般MySQL提供技术支持,但是一致性还是要用户业务逻辑做支撑,也就是,一致性,是由用户决定的。

        而技术上,通过AID保证C。

        RR级别的时候,多个事务的update,多个事务的insert,多个事务的delete,是会有加锁现象的,现象结果是,update,insert,delete之间是会有加锁现象的,但是select和这些操作是不冲突的。这就是通过读写锁(锁有行锁或者表锁)+MVCC完成隔离性。

        比如开启了两个事务,都对一个数据进行修改,会在事务每次commit时更新该数据,因此最后该数据的结果就是后commit的那个事务中修改的结果。

九.再次理解隔离性

1.数据库三种并发的场景

① 读-读 :不存在任何问题,也不需要并发控制

② 读-写 :有线程安全问题,可能会造成事务隔离性问题,可能遇到脏读,幻读,不可重复读

③ 写-写 :有线程安全问题,可能会存在更新丢失问题,比如第一类更新丢失,第二类更新丢失

2.读-写

(1)MVCC

        多版本并发控制( MVCC )是一种用来解决 读-写冲突 的无锁并发控制。

        为事务分配单向增长的事务ID,为每个修改保存一个版本,版本与事务ID关联,读操作只读该事务开始前的数据库的快照。

         所以 MVCC 可以为数据库解决以下问题:

        ① 在并发读写数据库时,可以做到在读操作时不用阻塞写操作,写操作也不用阻塞读操作,提高了数据库并发读写的性能

        ② 同时还可以解决脏读,幻读,不可重复读等事务隔离问题,但不能解决更新丢失问题

(2)3个隐藏列字段

        ① DB_TRX_ID :6 byte,最近修改( 修改/插入 )事务ID,记录创建这条记录/最后一次修改该记录的事务ID
        ② DB_ROLL_PTR : 7 byte,回滚指针,指向这条记录的上一个版本(简单理解成,指向历史版本就行,这些数据一般在 undo log 中)
        ③ DB_ROW_ID : 6 byte,隐含的自增ID(隐藏主键),如果数据表没有主键, InnoDB 会自动以DB_ROW_ID 产生一个聚簇索引

        ④ 补充:实际还有一个删除flag隐藏字段, 既记录被更新或删除并不代表真的删除,而是删除flag变了

        创建一个表:

        插入一个数据:

        显示的信息:

        这里有3个隐藏字段,是select显示不出来的,实际上显示是应该这样的:

ameageDB_TRX_ID(创建该记录的事
务ID)
DB_ROW_ID(隐式
主键)
DB_ROLL_PTR(回滚
指针)
张三28null1null

        我们目前并不知道创建该记录的事务ID,隐式主键,我们就默认设置成null,1。第一条记录也没有其他版本,我们设置回滚指针为null。

(3)undo日志

        MySQL 将来是以服务进程的方式,在内存中运行。我们之前所学的所有机制:索引,事务,隔离性,日志等,都是在内存中完成的,即在 MySQL 内部的相关缓冲区中,保存相关数据,完成各种判断操作。然后在合适的时候,将相关数据刷新到磁盘当中的。所以,我们这里理解undo log,简单理解成,就是 MySQL 中的一段内存缓冲区,用来保存日志数据的。

(4)模拟MVCC 

        现在有一个事务10,对student表中记录进行修改(update):将name(张三)改成name(李四)。

update student set name='李四' where name='张三';

       

        事务10,因为要修改,所以要先给该记录加行锁。

        修改前,先将改行记录拷贝到undo log中,所以,undo log中就有了一行副本数据。(原理就是写时拷贝)

        所以现在 MySQL 中有两行同样的记录。现在修改原始记录中的name,改成 '李四'。并且修改原始记录的隐藏字段 DB_TRX_ID 为当前 事务10 的ID, 我们默认从 10 开始,之后递增。而原始记录的回滚指针 DB_ROLL_PTR 列,里面写入undo log中副本数据的地址,从而指向副本记录,即表示我的上一个版本就是它。

        事务10提交,释放锁。

         

        现在又有一个事务11,对student表中记录进行修改(update):将age(28)改成age(38)。

update student set age=38 where name='李四';

         事务11,因为也要修改,所以要先给该记录加行锁。(该记录是那条?)

        修改前,现将改行记录拷贝到undo log中,所以,undo log中就又有了一行副本数据。此时,新的副本,我们采用头插方式,插入undo log。

        现在修改原始记录中的age,改成 38。并且修改原始记录的隐藏字段 DB_TRX_ID 为当前 事务11 的ID。而原始记录的回滚指针 DB_ROLL_PTR 列,里面写入undo log中副本数据的地址,从而指向副本记录,即表示我的上一个版本就是它。

        事务11提交,释放锁。

        单纯从回滚角度 :真实的做法,undo log里面保存相反的sql操作即可。

        这样,我们就有了一个基于链表记录的历史版本链。所谓的回滚,无非就是用历史数据,覆盖当前数据。

        上面的一个一个版本,我们可以称之为一个一个的快照。

        上面是以更新(`upadte`)主讲的,如果是`delete`呢?一样的,删数据不是清空,而是设置flag为删除即可,也可以形成版本链。

        如果是`insert`呢?因为`insert`是插入,也就是之前没有数据,那么`insert`也就没有历史版本。但是一般为了回滚操作,insert的数据也是要被放入undo log中,如果当前事务commit了,那么这个undolog 的历史insert记录就可以被清空了。

        总结一下,也就是我们可以理解成,`update`和`delete`可以形成版本链,`insert`暂时不考虑。那么`select`呢?

        首先,`select`不会对数据做任何修改,所以,为`select`维护多版本,没有意义。不过,此时有个问题,

        就是:
        select读取,是读取最新的版本呢?还是读取历史版本?

        当前读:读取最新的记录,就是当前读。增删改,都叫做当前读,select也有可能当前读,比如:selectlock in share mode(共享锁), select for update 
        快照读:读取历史版本(一般而言),就叫做快照读。

        我们可以看到,在多个事务同时删改查的时候,都是当前读,是要加锁的。那如果同时有select过来,也要读取最新版(当前读),那么也就需要加锁,这就是串行化。
        但如果是快照读,读取历史版本的话,是不受加锁限制的。也就是可以并行执行。换言之,提高了效率,即MVCC的意义所在。


        那么,是什么决定了,select是当前读,还是快照读呢?隔离级别。


        那为什么要有隔离级别呢?

        事务都是原子的。所以,无论如何,事务总有先有后。


        但是经过上面的操作我们发现,事务从begin->CURD->commit,是有一个阶段的。也就是事务有执行前,执行中,执行后的阶段。但,不管怎么启动多个事务,总是有先有后的。


        在那么多个事务执行中,CURD操作是会交织在一起的。那么,为了保证事务的“有先有后”,应该让不同的事务看到它该看到的内容,这就是所谓的隔离性与隔离级别要解决的问题。


        先来的事务,不应该看到后来的事务所做的修改。

(5)Read View

        当我们begin启动事务的时候,是没有read view的,但是肯定有事务ID和事务对象。

        当我们首次进行select的时候,mysqld会自动给我们形成read view。

        Read View就是事务进行 快照读 操作的时候生产的 读视图 (Read View),在该事务执行的快照读的那一刻,会生成数据库系统当前的一个快照,记录并维护系统当前活跃事务的ID(当每个事务开启时,都会被分配一个ID, 这个ID是递增的,所以最新的事务,ID值越大)

        Read View 在 MySQL 源码中,就是一个类,本质是用来进行可见性判断的。 即当我们某个事务执行快照读的时候,对该记录创建一个 Read View 读视图,把它比作条件,用来判断当前事务能够看到哪个版本的数据,既可能是当前最新的数据,也有可能是该行记录的 undo log 里面的某个版本的数据。

        ReadView结构:

class ReadView {
// 省略...

private:

        /** 高水位,大于等于这个ID的事务均不可见*/

        trx_id_t m_low_limit_id


        /** 低水位:小于这个ID的事务均可见 */
        trx_id_t m_up_limit_id;


        /** 创建该 Read View 的事务ID*/
        trx_id_t m_creator_trx_id;


        /** 创建视图时的活跃事务id列表*/
        ids_t m_ids;


        /** 配合purge,标识该视图不需要小于m_low_limit_no的UNDO LOG,
        * 如果其他视图也不需要,则可以删除小于m_low_limit_no的UNDO LOG*/
        trx_id_t m_low_limit_no;


        /** 标记视图是否被关闭*/
        bool m_closed;


        // 省略...
};

m_ids;                  //一张列表,用来维护Read View生成时刻,系统正活跃的事务ID
up_limit_id;          //记录m_ids列表中事务ID最小的ID
low_limit_id;         //ReadView生成时刻系统尚未分配的下一个事务ID,也就是目前已出现过                                  的事务ID的最大值+1

creator_trx_id;   //创建该ReadView的事务ID

        我们在实际读取数据版本链的时候,是能读取到每一个版本对应的事务ID的,即:当前记录的DB_TRX_ID 。

        那么,我们现在手里面有的东西就有,当前快照读的 ReadView 和 版本链中的某一个记录的DB_TRX_ID 。

        所以现在的问题就是,当前快照读,应不应该读到当前版本记录。

对应源码策略:

        如果查到不应该看到当前版本,接下来就是遍历下一个版本,直到符合条件,即可以看到。上面的readview 是当你进行select的时候,会自动形成。

        假设当前有条记录:

nameageDB_TRX_ID(创建该记录的事
务ID)
DB_ROW_ID(隐式
主键)
DB_ROLL_PTR(回滚
指针)
张三28null1null

        事务操作:

事务1 [id=1]事务2 [id=2]事务3 [id=3]事务4 [id=4]
事务开始事务开始事务开始事务开始
.........修改且已提交
进行中快照读进行中
.........

        事务4:修改name(张三) 变成name(李四)

        当 事务2 对某行数据执行了 快照读 ,数据库为该行数据生成一个 Read View 读视图

//事务2的 Read View
m_ids; // 1,3
up_limit_id; // 1
low_limit_id; // 4 + 1 = 5,原因:ReadView生成时刻,系统尚未分配的下一个事务ID
creator_trx_id // 2

        此时版本链是:

        只有事务4修改过该行记录,并在事务2执行快照读前,就提交了事务。

        我们的事务2在快照读该行记录的时候,就会拿该行记录的 DB_TRX_ID 去跟up_limit_id,low_limit_id和活跃事务ID列表(trx_list) 进行比较,判断当前事务2能看到该记录的版本。

//事务2的 Read View
m_ids; // 1,3
up_limit_id; // 1
low_limit_id; // 4 + 1 = 5,原因:ReadView生成时刻,系统尚未分配的下一个事务ID
creator_trx_id // 2
//事务4提交的记录对应的事务ID
DB_TRX_ID=4
//比较步骤
DB_TRX_ID(4)< up_limit_id(1) ? 不小于,下一步
DB_TRX_ID(4)>= low_limit_id(5) ? 不大于,下一步
m_ids.contains(DB_TRX_ID) ? 不包含,说明,事务4不在当前的活跃事务中。

        // 结论
        故,事务4的更改,应该看到。
        所以事务2能读到的最新数据记录是事务4所提交的版本,而事务4提交的版本也是全局角度上最新的版本

十.RR与RC的区别

1.当前读和快照读在RR级别下的区别:

select * from user lock in share mode ,以加共享锁方式进行读取,对应的就是当前读

        测试表:

--设置RR模式下测试

 

--重启终端
 

create table if not exists account(
id int primary key,
name varchar(50) not null default '',
blance decimal(10,2) not null default 0.0
)ENGINE=InnoDB DEFAULT CHARSET=UTF8;


insert into user (id, age, name) values (1, 28, '张三');

        测试1:

事务A操作事务A描
事务B描述事务B操作
begin开启事务开启事务begin
select * from
user
快照读
(无影响)
查询
快照读查询select * from user
update user set
age=18 where id=1;
更新
age=18
--
commit提交事务--
select 快照读 ,没有读到
age=18
select * from user
select lock in share
mode当前读 , 读到age=18
select * from user
lock in share mode

        测试2:

事务A操作事务A描
事务B描述事务B操作
begin开启事务开启事务begin
select * from
user
快照读,
查到
age=18
--
update user set
age=28 where id=1;
更新
age=28
--
commit提交事务--
select 快照读 age=28select * from user
select lock in share
mode当前读 age=28
select * from user
lock in share mode

        用例1与用例2:唯一区别仅仅是 表1的事务B在事务A修改age前 快照读 过一次age数据

        而表2的事务B在事务A修改age前没有进行过快照读。

结论:

        ① 事务中快照读的结果是非常依赖该事务首次出现快照读的地方,即某个事务中首次出现快照读,决定该事务后续快照读结果的能力
        ② delete同样如此

        几乎同时启动的事务,后者能否看到前者修改 && commit的信息,取决于后缀什么时候进行 快照读。快照读就会生成read view。

2.RR与RC的本质区别:

        正是Read View生成时机的不同,从而造成RC,RR级别下快照读的结果的不同

        在RR级别下的某个事务的对某条记录的第一次快照读会创建一个快照及Read View, 将当前系统活跃的其他事务记录起来

        此后在调用快照读的时候,还是使用的是同一个Read View,所以只要当前事务在其他事务提交更新之前使用过快照读,那么之后的快照读使用的都是同一个Read View,所以对之后的修改不可见

        即RR级别下,快照读生成Read View时,Read View会记录此时所有其他活动事务的快照,这些事务的修改对于当前事务都是不可见的。而早于Read View创建的事务所做的修改均是可见

        

        而在RC级别下的,事务中,每次快照读都会新生成一个快照和Read View, 这就是我们在RC级别下的事务中可以看到别的事务提交的更新的原因

        总之在RC隔离级别下,是每个快照读都会生成并获取最新的Read View;而在RR隔离级别下,则是同一个事务中的第一个快照读才会创建Read View, 之后的快照读获取的都是同一个Read View。

        

        正是RC每次快照读,都会形成Read View,所以,RC才会有不可重复读问题。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
MySQL 事务是指一组数据库操作,这些操作要么全部执行,要么全部不执行,其目的是保证在并发环境下,数据的一致性和完整性。MySQL 事务具有 ACID 性质,即原子性、一致性、隔离性和持久性。 MySQL 中使用事务需要使用 BEGIN、COMMIT 和 ROLLBACK 语句,其中 BEGIN 表示开启一个事务,COMMIT 表示提交事务,ROLLBACK 表示回滚事务事务的基本语法如下: ``` BEGIN; -- 执行一组数据库操作 COMMIT; -- 提交事务 -- 或者 ROLLBACK; -- 回滚事务 ``` 在 MySQL 中,事务的隔离级别分为四个等级,分别是 Read Uncommitted、Read Committed、Repeatable Read 和 Serializable。隔离级别越高,数据的一致性和完整性越高,但同时也会影响数据库的性能。 MySQL 事务的 ACID 性质有以下含义: 1. 原子性(Atomicity):事务中的所有操作要么全部执行成功,要么全部失败回滚,不会只执行其中的一部分操作。 2. 一致性(Consistency):事务执行前后,数据库中的数据必须保持一致性状态,即满足数据库的约束条件和完整性规则。 3. 隔离性(Isolation):事务之间应该是相互隔离的,一个事务的执行不应该被其他事务干扰,保证事务之间的数据相互独立。 4. 持久性(Durability):事务提交后,对数据库的修改应该是永久性的,即使出现系统故障或电源故障,也不应该对数据产生影响。 总之,MySQL 事务是一组数据库操作,具有 ACID 性质,可以通过 BEGIN、COMMIT 和 ROLLBACK 语句来实现,隔离级别越高,数据的一致性和完整性越高,但同时也会影响数据库的性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

冰果滴

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值