首页 > 数据库 >MySQL加锁逻辑与死锁

MySQL加锁逻辑与死锁

时间:2023-08-01 10:55:24浏览次数:40  
标签:10 加锁 sess1 死锁 MySQL where id delete

RC级别:

create table t1(id int primary key, name varchar(30));
insert into t1 values(1, 'a'),(4, 'c'),
(7, 'b'),(10, 'a'),(20, 'd'),(30, 'b');
commit;

案例一

--sess1           --sess2
begin;                              
delete from t1
where id = 10;
                 begin;
                 delete from t1 
                 where id < 9; ## 被阻塞

如果会话A,会话B的执行顺序调整一下,会话B 则不会被阻塞。

案例二


--sess1           --sess2
begin;                              
delete from t1
where id < 9;
                  begin;
                  delete from t1 
                  where id = 10;#不会阻塞

案例三 sess2 使用 delete from t1 where id < 9 order by id desc; 则没有阻塞。

--sess1          --sess2
begin;                              
delete from t1
where id = 10;
                 begin;
                 delete from t1 
                 where id < 9 order by id desc; ## 没有阻塞

为什么呢?

二 基础知识

sql被block住说明存在锁等待/锁冲突,等待其他会话释放锁。其实问题的核心在于 加锁顺序和加锁范围。这里结合 丁奇 《MySQL 实战45讲》中的讲述的 加锁方式(2个原则2个优化1个bug)

原则 1:加锁的基本单位是 next-key lock。

原则 2:查找过程中访问到的对象才会加锁。

优化 1:索引上的等值查询,给唯一索引加锁的时候,next-key lock 退化为行锁。

优化 2:索引上的等值查询,向右遍历时且最后一个值不满足等值条件的时候,next-key lock 退化为间隙锁。

一个 bug:唯一索引上的范围查询会访问到不满足条件的第一个值为止。

在读提交隔离级别下还有一个优化,即:语句执行过程中加上的行锁,在语句执行完成后,就要把“不满足条件的行”上的行锁直接释放了,不需要等到事务提交

三 案例分析

案例一

sess1 持有id = 10 该记录的行锁 lock_mode X locks rec but not gap

sess2 where 条件是 id<9 ,因为id=9 的记录不存在,根据原则 一个bug 故需要申请第一个不满足条件的记录id=10 的next-key ,也就是(7,10],此时id=10 的行锁被sess1 持有,故sess2产生锁等待被block住。

案例二

sess1 先执行delete t1 where id<9 ,根据RC模式的加锁方式 ,虽然要获取到第一个不满足记录的id=10 的next-key lock ,但是因为id=10 不符合 id<9 的条件,会释放锁,最终降级为 (7,10) 之间的gap lock。

再解释一下:这个过滤操作是 MySQL Sever层做的,也就是 innodb层把包括id=10的记录加锁然后发送给MySQL Server层,然后 MySQL Sever层判断是否where条件可以结束了,可以结束了则将不满足条件的id=10解锁。

sess2 delete where id=10 和 sess1 持有的gap lock不冲突,故可以顺利执行。

案例三

sess1 持有id = 10 该记录的行锁 lock_mode X locks rec but not gap

sess2 where 条件是 id<9 order by id desc ,通过innodb api接口访问数据的时候从获取到第一个满足条件的记录是id=7,不会访问 id=10这条记录。故也不会加上id=10 的next-key lock. 不会与sess1持有的锁冲突。

转载自杨奇龙博客

标签:10,加锁,sess1,死锁,MySQL,where,id,delete
From: https://www.cnblogs.com/lovezhr/p/17595907.html

相关文章

  • RR 死锁一侧
    CREATE TABLE `t8` (  `id` bigint(20) NOT NULL AUTO_INCREMENT,  `d_id` varchar(40) NOT NULL DEFAULT '',  `b_id` varchar(40) NOT NULL DEFAULT '',  `is_dropped` tinyint(1) NOT NULL DEFAULT '0',  `u_c` varchar(10) NOT NULL DEFA......
  • Java面试题 P28:数据库篇:MySql篇-MySql优化-索引-什么是索引?索引的底层数据结构是什么?
    什么是索引:索引(index)是帮助MySql高效获取数据的数据结构(有序)。在数据之外,数据库还维护着满足特定查找算法的数据结构(B+树),这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构上实现高级查找算法,这种数据结构就是索引。 ......
  • 查看mysql有无启动日志
    showvariableslike'log_%'这些参数代表着MySQL的日志相关配置,对每个配置进行介绍如下:1.`log_bin`:指定二进制日志文件的开启状态。2.`log_bin_basename`:指定二进制日志文件名的前缀,MySQL会在其后面自动增加数字序列作为文件名。3.`log_bin_index`:指定MySQL的二进制......
  • MySQL 巨坑:永远不要在 MySQL 中使用 UTF-8!!请使用utf8mb4
    https://mp.weixin.qq.com/s?__biz=MzU4NDc1NDMxMw==&mid=2247486462&idx=1&sn=7c490f075e3b7d704ae8415a6afd3c30&chksm=fd95be18cae2370e4001eada1c3e33efe6dd5fb8afd5024aa17d001f8609b2c4d821a03280c4&mpshare=1&scene=24&srcid=0731Pfqoe5Yw......
  • Java面试题 P27:数据库篇:MySql篇-MySql优化-Sql语句执行很慢,如何分析呢?
       ......
  • Java面试题 P26:数据库篇:MySql篇-MySql优化-如何定位慢查询?
          ......
  • 【已解决】如果将MySQL数据库中的表生成PDM
    数据库表PDM关系图                        | 原创作者/编辑:凯哥Java                                    | 分类:经验分享有时候,我们需要MySQL数据库中的表生成对应的PDM文件,这里凯哥就讲讲第一种将MySQL数据库......
  • 【已解决】如果将MySQL数据库中的表生成PDM
     数据库表PDM关系图                                                        | 原创作者/编辑:凯哥Java                                     | 分类:经验分享 有......
  • mysql优化--索引
    mysql优化--索引Mysql索引大概有五种类型:普通索引(INDEX):最基本的索引,没有任何限制唯一索引(UNIQUE):与"普通索引"类似,不同的就是:索引列的值必须唯一,但允许有空值。主键索引(PRIMARY):它是一种特殊的唯一索引,不允许有空值。全文索引(FULLTEXT):可用于MyISAM表,mysql5.6之后也可......
  • 6、Mysql操作数据库以及数据表
    学习sql规则,可以让mysql服务器帮咱们做其他操作1、操作数据库(文件夹)createdatabase数据库名defaultcharsetutf8;表示整个数据库是utf8的格式 use数据库名;使用这个数据库 查看数据库showdatabases; 删除数据库dropdatabase数据库名;数据库没有修改这一说......