博客
关于我
Mysql-解决幻读
阅读量:436 次
发布时间:2019-03-06

本文共 1821 字,大约阅读时间需要 6 分钟。

2.MVCC和Next-key locks

2.1 MVCC(MultiVersion Concurrent Control)

2.1.1 机制

MVCC即多版本并发控制,是InnoDB自带的一种机制,其为表格添加了三个列,如下

DB_ROW_ID DB_ROLL_PTR DB_TRX_ID col1
隐藏的行id,当没有合适的列可以创建索引时,使用此列创建索引 回滚指针,指向Undo Log中的上一版本数据的地址 记录当前事务的id

其中DB_TRX_ID和DB_ROLL_PTR是关键,下面举例说明,

当一个事务A开启后,在RR隔离级别下,只在第一次调用select时会获得一个表格的Read View,其包括四个属性

  • trx_ids: 截至目前,系统活跃的事务版本号,包括自己
  • low_limit_id: 当前事务版本号+1
  • up_limit_id: 除本事务外,截至目前,活跃事务的最小版本号
  • creator_trx_id: 创建当前read view的事务版本号

那么在开启事务A后,执行

  • SELECT

    会根据Read View对数据进行过滤,其可以读取到的数据有如下部分

    • 本事务内添加的数据
    • 所有版本号小于up_limit_id的已提交事务的结果

    除上面提到的,都不可以select到,如trx_ids中除了当前事务号插入的数据,因为这些事务仍然活跃且未提交

  • 包括update、insert、delete,都会将对应数据行的DB_TRX_ID赋值为当前事务版本号

2.1.2 当前读和快照读

1)快照读

MVCC机制下,select读取数据是不需要加锁的,其读取的是快照中的数据,包含事务开启前的数据以及当前事务添加的数据

2)当前读

也即insert、update、delete这三种方式,其会对访问数据加锁

见下面例子

事务A 事务B
begin;
begin;
select * from test; 🎈
insert into test(id,name) values(3,'kk');
select * from test;与🎈处查询结果一致
commit;
insert into test(id,name) values(3,'kk'); 报错!因为此数值已经出现在表格中,但是由于查询时遵循MVCC的机制,因而不能查询出未提交事务的结果;如果在事务Bcommit之前发生,则会被阻塞
select * from test;//可以查询到事务B插入的值

从上可见,由于insert操作属于当前读,事务A在Bcommit前进行insert时,会被阻塞,但是select不会被阻塞

那么,在事务A中进行insert这个当前读操作就发生了幻读,因为这和select读取的结果不一致

2.2 Next-Key Locks

从MVCC的机制可知,其不能解决幻读,InnoDB引入了Next-Key Lock解决了此问题(关键在于上锁

2.2.1 Record Locks

锁定一个记录上的索引,而不是记录本身

2.2.2 Gap Locks

锁定索引之间的间隙

假设select * from test where id between 10 and 20 for update;

那么id从10到20的数据就会被锁住,尽管10-20之间可能没有数据

2.2.3 Next-Key Locks

首先会对指定的行的索引上锁,且会对索引之前所在的gap上锁

假设在事务A中执行select * from test where id=3 for update;且假设当前表格中有id为1,5,10

那么,有如下情况,如果id为

  • 主键(UNIQUE

    不会触发gap lock

  • 普通索引(UNUNIQUE

    会锁住id为(1,5]的区间

  • 不是索引

    那么会对(-∞,+∞)上锁

2.3 Serializable解决幻读

MySQL下的这个隔离级别解决幻读的方式很悲观,其会对事务中每条sql涉及到的资源进行上锁,这样就可以阻塞其他事务对数据的操作,从而使得事务可以串行化的执行

可见,这种方式的效率是很低的

一般情况下,在RR隔离级别时,可以通过使用Next-Key Locks的方式对敏感数据上锁,即可保证一定区域内幻读的解决

触发Next-Key Locks有select for update和当前读操作

# 参考

转载地址:http://kasyz.baihongyu.com/

你可能感兴趣的文章
MySql中给视图添加注释怎么添加_默认不支持_可以这样取巧---MySql工作笔记002
查看>>
Mysql中获取所有表名以及表名带时间字符串使用BetweenAnd筛选区间范围
查看>>
Mysql中视图的使用以及常见运算符的使用示例和优先级
查看>>
Mysql中触发器的使用示例
查看>>
Mysql中设置只允许指定ip能连接访问(可视化工具的方式)
查看>>
mysql中还有窗口函数?这是什么东西?
查看>>
mysql中间件
查看>>
MYSQL中频繁的乱码问题终极解决
查看>>
MySQL为Null会导致5个问题,个个致命!
查看>>
MySQL为什么不建议使用delete删除数据?
查看>>
MySQL主从、环境搭建、主从配制
查看>>
Mysql主从不同步
查看>>
mysql主从同步及清除信息
查看>>
MySQL主从同步相关-主从多久的延迟?
查看>>
mysql主从同步配置方法和原理
查看>>
mysql主从复制 master和slave配置的参数大全
查看>>
MySQL主从复制几个重要的启动选项
查看>>
MySQL主从复制及排错
查看>>
mysql主从复制及故障修复
查看>>
MySQL主从复制的原理和实践操作
查看>>