MySQL(2):事务的隔离级别及其原理(MVCC)

2023-10-31  本文已影响0人  hcq0514

在讲隔离级别之前先了解一下并发事务会带来的问题

事务的隔离级别

事务的隔离级别实现原理(MVCC)

多版本并发控制(MVCC,Multi-Version Concurrency Control)是一种数据库管理系统用于实现不同隔离级别的并发性控制机制。MVCC通过在数据库中保留不同版本的数据(undoLog)来允许多个事务同时操作数据库而不相互干扰。每个事务只能看到适合其隔离级别的数据版本,从而确保数据的隔离和一致性。以下是MVCC的详细解释:

MVCC的实现原理(基于undo_log)

MVCC和undo日志之间的关系在于,undo日志用于保存原始数据的版本,以便在事务需要回滚或其他事务需要访问旧版本数据时使用。
这些undo日志记录与MVCC的时间戳机制协同工作,以确保事务可以看到适当版本的数据,并支持隔离级别的实现。
mvcc的那些旧版本数据都是在undoLog里面查询的,当是可重复读的时候需要生成一份快照,其实没有重新创建一份数据,只是保存了当前事务开启的事务id,然后查询undolog的时候去判断在这个事务id之前的读取到,在这个事务之后的相当于后来的,就不会读取了

MVCC实现各个隔离级别的方式

随着隔离级别的提高,数据版本的选择和可见性受到更严格的控制,从而确保了数据的一致性和隔离性。不同数据库管理系统的具体实现方式可能有所不同,但核心概念是使用时间戳和事务的启动时间戳来确定事务可以看到的数据版本范围。

既然可重复读是使用mvcc来做快照的,为什么会有幻读的问题呢

理论上mvcc读取的是快照,不会读取到后面修改的数据,但是这里面有两个概念,一个是快照读,一个是当前读

  1. 快照读 简单的查询select读取的是快照读
  1. 当前读,当涉及到,insert,update,delete,(for update,for share等),这些语句在执行前都会现查询一下,这时候就会使用当前读,当前这条语句会去导致快早更新为读取数据库里的最新数据,
所以结论是:如果语句没有insert,update,delete,for update,insert等会导致当前读的情况下,其实可重复读也是可以保证没有幻读的情况出现,但是在有这些语句的时候就没有办法了
insert幻读示例:
事务A 事务A 事务A
begin begin
select * from account where id = 2; Empty set (0.00 sec) 没有数据
INSERT INTO account(id, name) VALUES (2, 'n2');commit; 此时事务B插入数据并提交事务
select * from account where id = 2; Empty set (0.00 sec) 没有数据
INSERT INTO account(id, name) VALUES (2, '23'); 事务A执行插入一条id为2的数据
ERROR 1062 (23000): Duplicate entry '2' for key 'PRIMARY' 插入报错,读取到了事务B插入的数据
update幻读示例:
事务A 事务A 事务A
begin begin
selct balance from account where id = 1; 此时balance为0
update account set balance =100 where id =1; commit 此时事务B修改balance为100并且提交事务
selct balance from account where id = 1; 此时事务A查询结果还是0
update account set balance = balance+100 where id =1; 此时事务A更新balance
selct balance from account where id = 1; 此时事务A查询结果是200,出现幻读
上一篇 下一篇

猜你喜欢

热点阅读