MySql@花城鱼

MySQL InnoDB 并发控制,事务的实现 学习笔记

2017-02-18  本文已影响0人  专职跑龙套

所在文集:数据库


如何保证数据一致性

通过并发控制保证数据一致性的常见手段有:

如何使用普通锁保证一致性?

普通锁存在什么问题?
简单的锁住太过粗暴,连“读任务”也无法并行,任务执行过程本质上是串行的。

于是出现了共享锁排他锁

有没有可能,进一步提高并发呢?
即使写任务没有完成,其他读任务也可能并发,这就引出了数据多版本
数据多版本是一种能够进一步提高并发的方法,它的核心原理是:

数据多版本

如上图:

可以看到,数据多版本,通过“读取旧版本数据”能够极大提高任务的并发度。

提高并发的演进思路,就在如此:

undo log

日志文件,记录数据被 修改前 的值,用来进行 rollback回滚。
如某个事务 T1,将 X 的值由 5 修改为 10,则 undo log 写入 <T1,X,5>
对于 insert 操作,undo log 记录新数据的 PK(ROW_ID),回滚时直接删除;

为什么要有 undo log?
数据库事务未提交时,会将事务修改数据的镜像(即修改前的旧版本)存放到 undo log 里,当事务回滚时,或者数据库奔溃时,可以利用 undo log,即旧版本数据,撤销未提交事务对数据库产生的影响。

一句话,undo日志用于保障,未提交事务不会对数据库的ACID特性产生影响。

磁盘上 不存在 单独的 undo log 文件,所有的 undo log 均存放在主 ibd 数据文件中(表空间)。
记录先写入到 undo buffer,但当缓冲满的时候,undo buffer 中的内容会也会被刷新到磁盘。

redo log

日志文件,记录数据被 修改后 的值,回来恢复未写入到磁盘文件的已成功事务更新的数据。
如某个事务 T1,将 X 的值由 5 修改为 10,则 redo log 写入 <T1,X,10>

为什么要有 redo log?
数据库事务提交后,必须将更新后的数据刷到磁盘上,以保证 ACID 特性。磁盘随机写性能较低,如果每次都刷盘,会极大影响数据库的吞吐量。
优化方式是,将修改行为先写到 redo log 里,再定期将数据刷到磁盘上,这样能极大提高性能。
假如某一时刻,数据库崩溃,还没来得及刷盘的数据,在数据库重启后,会重做 redo log 日志里的内容,以保证已提交事务对数据产生的影响都刷到磁盘上。

一句话,redo log 用于保障,已提交事务的 ACID 特性。

磁盘上 存在 单独的 redo log 文件。
记录先写入到 redo buffer,但当缓冲满的时候,redo buffer 中的内容会也会被刷新到磁盘。

rollback segment 回滚段

回滚段。在 InnoDB 中,undo log 被划分为多个段,具体某行的 undo log 就保存在某个段中,称为回滚段。

事务的过程

假设 User 表有两个字段 <name, salary>
InnoDB 为每行记录都实现了三个隐藏字段:

因此实际的 User 表有五个字段 <name, salary, ID, DB_TRX_ID, DB_ROLL_PTR>

假设某一行的初始值为:<Tom, 10000, 1, NULL, NULL>

假设某一个事务里执行了 update users set salary = 20000 where name = 'Tom',存储引擎会依次进行如下操作:

若 提交了 事务
只需要更改事务状态为 COMMIT 即可,不需做其他额外的工作。

若 回滚了 事务
需要根据当前 DB_ROLL_PTR 回滚指针 从 undo log 中找出事务修改前的版本,并恢复。

一个具体的示例

数据表 t(id PK, name)
数据为:

1, shenjian
2, zhangsan
3, lisi

此时没有事务未提交,故回滚段是空的。

接着启动了一个事务,并且事务处于未提交的状态。

start trx;
delete (1, shenjian);
update set(3, lisi) to (3, xxx);
insert (4, wangwu);

可以看到:

接下来,假如事务 rollback,此时可以通过回滚段里的 undo log 回滚。



可以看到:

InnoDB 是基于多版本并发控制的存储引擎

InnoDB 是高并发互联网场景最为推荐的存储引擎,根本原因,就是其多版本并发控制(Multi Version Concurrency Control, MVCC)。行锁,并发,事务回滚等多种特性都和 MVCC 相关。
MVCC就是通过“读取旧版本数据”来降低并发事务的锁冲突,提高任务的并发度。

InnoDB为何能够做到这么高的并发?
回滚段里的数据,其实是历史数据的快照(snapshot),这些数据是不会被修改,select 可以肆无忌惮的并发读取他们。

快照读(Snapshot Read),这种一致性不加锁的读(Consistent Nonlocking Read),就是 InnoDB 并发如此之高的核心原因之一。

这里的一致性是指,事务读取到的数据,要么是事务开始前就已经存在的数据(当然,是其他已提交事务产生的),要么是事务自身插入或者修改的数据。

什么样的 select 是快照读?
除非显示加锁,普通的select语句都是快照读,例如:
select * from t where id>2;

这里的显示加锁,非快照读是指:
select * from t where id>2 lock in share mode;
select * from t where id>2 for update;

总结:


引用:
Mysql InnoDB MVCC 实现原理
InnoDB并发如此高,原因竟然在这?

上一篇下一篇

猜你喜欢

热点阅读