前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >RR与RC隔离级别下MySQL不同的加锁解锁方式

RR与RC隔离级别下MySQL不同的加锁解锁方式

原创
作者头像
沃趣科技
发布2018-05-15 17:53:16
2.4K0
发布2018-05-15 17:53:16
举报
文章被收录于专栏:沃趣科技沃趣科技

|  RC与RR隔离级别下MySQL不同的加锁解锁方式


  • MySQL5.7.21
  • 数据准备
代码语言:javascript
复制
root@localhost : pxs 05:26:27> show create table dots\G
*************************** 1. row ***************************
  Table: dots
Create Table: CREATE TABLE `dots` (
`id` int(11) NOT NULL,
`color` varchar(20) COLLATE utf8_bin NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin
1 row in set (0.00 sec)
root@localhost : pxs 05:27:34> select * from dots;
+----+-------+
| id | color |
+----+-------+
|  1 | black |
|  2 | white |
|  3 | black |
|  4 | white |
+----+-------+
4 rows in set (0.00 sec)
root@localhost : pxs 01:57:02> show variables like 'innodb_locks_unsafe_for_binlog';
+--------------------------------+-------+
| Variable_name                  | Value |
+--------------------------------+-------+
| innodb_locks_unsafe_for_binlog | OFF  |
+--------------------------------+-------+
1 row in set (0.00 sec)

1.RC隔离级别

  • 确认隔离级别
代码语言:javascript
复制
root@localhost : pxs 05:27:35> show variables like '%iso%';
+-----------------------+----------------+
| Variable_name        | Value          |
+-----------------------+----------------+
| transaction_isolation | READ-COMMITTED |
| tx_isolation          | READ-COMMITTED |
+-----------------------+----------------+
2 rows in set (0.01 sec)
  • 同时开启两个会话,按下图的流程开始操作。

2.RR隔离级别

  • 确认隔离级别
代码语言:javascript
复制
root@localhost : pxs 05:24:41> show variables like '%iso%';
+-----------------------+-----------------+
| Variable_name        | Value          |
+-----------------------+-----------------+
| transaction_isolation | REPEATABLE-READ |
| tx_isolation          | REPEATABLE-READ |
+-----------------------+-----------------+
2 rows in set (0.01 sec)
  • 同时开启两个会话,按下图的流程开始操作。

3.半一致读semi-consistent read

3.1 半一致读发生条件

  • RC隔离级别
  • RR隔离级别,且innodb_locks_unsafe_for_binlog=true

3.2 innodb_locks_unsafe_for_binlog

  • innodb_locks_unsafe_for_binlog默认为off。 
  • 如果设置为1,会禁用gap锁,但对于外键冲突检测(foreign-key constraint checking)或者重复键检测(duplicate-key checking)还是会用到gap锁。  
  • 启用innodb_locks_unsafe_for_binlog产生的影响等同于将隔离级别设置为RC,不同之处是:

1)innodb_locks_unsafe_for_binlog是全局参数,影响所有session;但隔离级别可以是全局也可以是会话级别。

2)innodb_locks_unsafe_for_binlog只能在数据库启动的时候设置;但隔离级别可以随时更改。    基于上述原因,RC相比于innodb_locks_unsafe_for_binlog会更好更灵活。

启用innodb_locks_unsafe_for_binlog还有以下作用:

  • 对于update或者delete语句,InnoDB只会持有匹配条件的记录的锁。在MySQL Server过滤where条件,发现不满足后,会把不满足条件的记录释放锁。这可以大幅降低死锁发生的概率。 
  • 简单来说,semi-consistent read是read committed与consistent read两者的结合。一个update语句,如果读到一行已经加锁的记录,此时InnoDB返回记录最近提交的版本,由MySQL上层判断此版本是否满足update的where条件。若满足(需要更新),则MySQL会重新发起一次读操作,此时会读取行的最新版本(并加锁)。

来看下面这个例子:

代码语言:javascript
复制
CREATE TABLE t (a INT NOT NULL, b INT) ENGINE = InnoDB; 
INSERT INTO t VALUES (1,2),(2,3),(3,2),(4,3),(5,2); 
COMMIT;

这个例子中,表上没有索引,所以对于记录锁会用到隐藏主键。

假设某个client开启了一个update:

代码语言:javascript
复制
SET autocommit = 0; 
UPDATE t SET b = 5 WHERE b = 3;

假设另一个client紧接着也开启一个update:

代码语言:javascript
复制
SET autocommit = 0; 
UPDATE t SET b = 4 WHERE b = 2;

每当InnoDB发起update,会先对每一行记录加上排它锁,然后再决定记录是否满足条件。如果不匹配,则innodb_locks_unsafe_for_binlog开启,InnoDB就会把记录上的锁释放掉。否则,InnoDB会一直持有锁直到事务结束。具体如下:

如果innodb_locks_unsafe_for_binlog没有开启,第一个update会一直持有x锁

代码语言:javascript
复制
x-lock(1,2); retain x-lock 
x-lock(2,3); update(2,3) to (2,5); retain x-lock 
x-lock(3,2); retain x-lock 
x-lock(4,3); update(4,3) to (4,5); retain x-lock 
x-lock(5,2); retain x-lock

第二个update会阻塞住直到第一个update提交或者回滚

代码语言:javascript
复制
x-lock(1,2); block and wait for first UPDATE to commit or roll back

如果innodb_locks_unsafe_for_binlog开启,第一个update先持有x锁,然后会释放不匹配的记录上面的x锁

代码语言:javascript
复制
x-lock(1,2); unlock(1,2) 
x-lock(2,3); update(2,3) to (2,5); retain x-lock 
x-lock(3,2); unlock(3,2) 
x-lock(4,3); update(4,3) to (4,5); retain x-lock 
x-lock(5,2); unlock(5,2)

对于第二个update,InnoDB会开启半一致读,此时InnoDB返回记录最近提交的版本,由MySQL上层判断此版本是否满足update的where条件。

代码语言:javascript
复制
x-lock(1,2); update(1,2) to (1,4); retain x-lock 
x-lock(2,3); unlock(2,3) 
x-lock(3,2); update(3,2) to (3,4); retain x-lock 
x-lock(4,3); unlock(4,3) 
x-lock(5,2); update(5,2) to (5,4); retain x-lock

4.一开始的例子

4.1 RC隔离级别

session 1

session 1执行:

代码语言:javascript
复制
update dots set color = 'black' where color = 'white'; 

由于color列无索引,因此只能走聚簇索引,进行全部扫描。加锁如下: 

注:如果一个条件无法通过索引快速过滤,那么存储引擎层面就会将所有记录加锁后返回,然后由MySQL Server层进行过滤。因此也就把所有的记录,都锁上了。

但在实际中,MySQL做了优化,如同前面作用1所提到的。在MySQL Server过滤条件,发现不满足后,会调用unlock_row方法,把不满足条件的记录放锁 (违背了2PL的约束)。这样做,保证了最后只会持有满足条件记录上的锁,但是每条记录的加锁操作还是不能省略的。 

实际加锁如下: 

session 2

session 2执行:

代码语言:javascript
复制
update dots set color = 'white' where color = 'black'; 

session 2尝试加锁的时候,发现行上已经存在锁,InnoDB会开启semi-consistent read,返回最新的committed版本(1,black),(2,white),(3,black),(4,white)。MySQL会重新发起一次读操作,此时会读取行的最新版本(并加锁)。如同前面作用2所提到的。 

加锁如下: 

MySQL优化后实际加锁如下: 

4.2 RR隔离级别

session 1

session 1执行:

代码语言:javascript
复制
update dots set color = 'black' where color = 'white'; 

由于color列无索引,因此只能走聚簇索引,进行全部扫描。加锁如下: 

session 2

session 2执行:

代码语言:javascript
复制
update dots set color = 'white' where color = 'black';

更新被阻塞。  等session 1提交commit之后,session 2update才会成功。

引申:RR隔离级别,且开启innodb_locks_unsafe_for_binlog=ON

  • 环境准备
代码语言:javascript
复制
root@localhost : (none) 04:57:46> show  variables like '%iso%';
+-----------------------+-----------------+
| Variable_name        | Value          |
+-----------------------+-----------------+
| transaction_isolation | REPEATABLE-READ |
| tx_isolation          | REPEATABLE-READ |
+-----------------------+-----------------+
2 rows in set (0.01 sec)
root@localhost : (none) 04:55:25> show variables like 'innodb_locks_unsafe_for_binlog';
+--------------------------------+-------+
| Variable_name                  | Value |
+--------------------------------+-------+
| innodb_locks_unsafe_for_binlog | ON    |
+--------------------------------+-------+
1 row in set (0.00 sec)
root@localhost : pxs 05:00:54> select * from dots;
+----+-------+
| id | color |
+----+-------+
|  1 | black |
|  2 | white |
|  3 | black |
|  4 | white |
+----+-------+
4 rows in set (0.00 sec)
  • 开始操作

注:过程现象满足RR隔离级别,也符合设置innodb_locks_unsafe_for_binlog=ON的情况。因为前面所讲的启用innodb_locks_unsafe_for_binlog会产生作用1与作用2,所以整个加锁与解锁情况与RC隔离级别类似。

参考:

《数据库事务处理的艺术:事务管理与并发控制》  https://dev.mysql.com/doc/refman/5.5/en/innodb-parameters.html#sysvar_innodb_locks_unsafe_for_binlog http://hedengcheng.com/?p=771 http://hedengcheng.com/?p=220

|  作者简介

韩杰  沃趣科技MySQL数据库工程师

熟悉mysql体系架构、主从复制,熟悉问题定位与解决。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • |  RC与RR隔离级别下MySQL不同的加锁解锁方式
    • 1.RC隔离级别
      • 2.RR隔离级别
        • 3.半一致读semi-consistent read
          • 4.一开始的例子
          • 参考:
          • |  作者简介
          相关产品与服务
          云数据库 SQL Server
          腾讯云数据库 SQL Server (TencentDB for SQL Server)是业界最常用的商用数据库之一,对基于 Windows 架构的应用程序具有完美的支持。TencentDB for SQL Server 拥有微软正版授权,可持续为用户提供最新的功能,避免未授权使用软件的风险。具有即开即用、稳定可靠、安全运行、弹性扩缩等特点。
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档