♥️作者:小刘在C站
♥️个人主页: 小刘主页
♥️努力不一定有回报,但一定会有收获加油!一起努力,共赴美好人生!
♥️学习两年总结出的运维经验,以及思科模拟器全套网络实验教程。专栏:云计算技术
♥️小刘私信可以随便问,只要会绝不吝啬,感谢CSDN让你我相遇!
前言
上章讲到InnoDB引擎 (上)本章继续链接InnoDB引擎 (上)
目录
3 事务原理
3.1 事务基础
1). 事务
2). 特性
3.2 redo log
3.3 undo log
4 MVCC
4.1 基本概念
1). 当前读
2). 快照读
3). MVCC
4.2 隐藏字段
4.2.1 介绍
4.2.2 测试
1). 查看有主键的表 stu
2). 查看没有主键的表 employee
而对于这四大特性,实际上分为两个部分。 其中的原子性、一致性、持久化,实际上是由 InnoDB 中的两份日志来保证的,一份是 redo log 日志,一份是 undo log 日志。 而持久性是通过数据库的锁, 加上 MVCC 来保证的。
我们在讲解事务原理的时候,主要就是来研究一下redolog,undolog以及MVCC。
我们知道,在 InnoDB 引擎中的内存结构中,主要的内存区域就是缓冲池,在缓冲池中缓存了很多的数据页。 当我们在一个事务中,执行多个增删改的操作时, InnoDB 引擎会先操作缓冲池中的数据,如果缓冲区没有对应的数据,会通过后台线程将磁盘中的数据加载出来,存放在缓冲区中,然后将缓冲池中的数据修改,修改后的数据页我们称为脏页。 而脏页则会在一定的时机,通过后台线程刷新到磁盘中,从而保证缓冲区与磁盘的数据一致。 而缓冲区的脏页数据并不是实时刷新的,而是一段时间之后将缓冲区的数据刷新到磁盘中,假如刷新到磁盘的过程出错了,而提示给用户事务提交成功,而数据却没有持久化下来,这就出现问题了,没有保证事务的持久性。
那么,如何解决上述的问题呢? 在 InnoDB 中提供了一份日志 redo log ,接下来我们再来分析一 下,通过 redolog 如何解决这个问题。
有了 redolog 之后,当对缓冲区的数据进行增删改之后,会首先将操作的数据页的变化,记录在 redolog buffer 中。在事务提交时,会将 redo log buffer 中的数据刷新到 redo log 磁盘文件中。 过一段时间之后,如果刷新缓冲区的脏页到磁盘时,发生错误,此时就可以借助于 redo log 进行数据恢复,这样就保证了事务的持久性。 而如果脏页成功刷新到磁盘 或 或者涉及到的数据已经落盘,此时 redolog 就没有作用了,就可以删除了,所以存在的两个 redolog 文件是循环写的。 那为什么每一次提交事务,要刷新 redo log 到磁盘中呢,而不是直接将 buffer pool 中的脏页刷新 到磁盘呢 ? 因为在业务操作中,我们操作数据一般都是随机读写磁盘的,而不是顺序读写磁盘。 而 redo log 在 往磁盘文件中写入数据,由于是日志文件,所以都是顺序写的。顺序写的效率,要远大于随机写。 这 种先写日志的方式,称之为 WAL ( Write-Ahead Logging )。
在测试中我们可以看到,即使是在默认的 RR 隔离级别下,事务 A 中依然可以读取到事务 B 最新提交的内容,因为在查询语句后面加上了 lock in share mode 共享锁,此时是当前读操作。当然,当我们 加排他锁的时候,也是当前读操作。
在测试中 , 我们看到即使事务 B 提交了数据 , 事务 A 中也查询不到。 原因就是因为普通的 select 是快照 读,而在当前默认的 RR 隔离级别下,开启事务后第一个 select 语句才是快照读的地方,后面执行相同的 select 语句都是从快照中获取数据,可能不是当前的最新数据,这样也就保证了可重复读。
当我们创建了上面的这张表,我们在查看表结构的时候,就可以显式的看到这三个字段。 实际上除了这三个字段以外, InnoDB 还会自动的给我们添加三个隐藏字段及其含义分别是:
而上述的前两个字段是肯定会添加的, 是否添加最后一个字段 DB_ROW_ID ,得看当前表有没有主键,如果有主键,则不会添加该隐藏字段。
ibd2sdi stu.ibd查看到的表结构信息中,有一栏 columns ,在其中我们会看到处理我们建表时指定的字段以外,还有额外的两个字段 分别是: DB_TRX_ID 、 DB_ROLL_PTR ,因为该表有主键,所以没有 DB_ROW_ID隐藏字段。
create table employee (id int , name varchar(10));此时,我们再通过以下指令来查看表结构及其其中的字段信息:
ibd2sdi employee.ibd查看到的表结构信息中,有一栏 columns ,在其中我们会看到处理我们建表时指定的字段以外,还有额外的三个字段 分别是: DB_TRX_ID 、 DB_ROLL_PTR 、 DB_ROW_ID ,因为 employee 表是没有指定主键的。
♥️关注,就是我创作的动力
♥️点赞,就是对我最大的认可
♥️这里是小刘,励志用心做好每一篇文章,谢谢大家