40 | insert语句的锁为什么这么多?
40 | insert语句的锁为什么这么多?
讲述:林晓斌
时长12:18大小11.25M
insert … select 语句
insert 循环写入
insert 唯一键冲突
insert into … on duplicate key update
小结
上期问题时间
赞 38
提建议
精选留言(83)
- huolang2019-02-13老师,死锁的例子,关于sessionA拿到的c=5的记录锁,sessionB和sessionC发现唯一键冲突会加上读锁我有几个疑惑: 1. sessionA拿到的c=5的记录锁是写锁吗? 2. 为什么sessionB和sessionC发现唯一键冲突会加上读锁? 3. 如果sessionA拿到c=5的记录所是写锁,那为什么sessionB和sessionC还能加c=5的读锁,写锁和读锁不应该是互斥的吗? 4. sessionA还没有提交,为什么sessionB和sessionC能发现唯一键冲突?展开
作者回复: 1. 是的 2. 这个我觉得是为了防止这个记录再被删除(不过这个理由不是很硬,我还没有找到其他解释 3. 互斥的,所以这两个语句都在等待。注意next-key lock是由间隙锁和记录锁组成的哦, 间隙锁加成功了的。好问题。 4. 还没有提交,但是这个记录已经作为最新记录写进去了,复习一下08篇哈
共 23 条评论100 - 轻松的鱼2019-03-06老师好,想请教一下死锁的例子中: 1. 在 session A rollback 前,session B/C 都因为唯一性冲突申请了 S Next-key lock,但是被 session A 的 X but not gap lock 阻塞; 2. 在 session A rollbak 后,session B/C 顺利获得 S Next-key lock,并且都要继续进行插入,这时候我认为是因为插入意向锁(LOCK_INSERT_INTENTION)导致的死锁,因为插入意向锁会被 gap lock 阻塞,造成了相互等待。还没有进入到记录 X lock。 不知道我分析的对不对?展开
作者回复: 对
共 11 条评论66 - sonic2019-02-14你好, 我想问下文章中关于为什么需要创建临时表有这一句话: 如果读出来的数据直接写回原表,就可能在遍历过程中,读到刚刚插入的记录,新插入的记录如果参与计算逻辑,就跟语义不符。 我的疑问是:既然隔离级别是可重复读,照理来说新插入的的记录应该不会参与计算逻辑呀。展开
作者回复: 可重复读隔离级别下,事务是可以看到自己刚刚修改的数据的 ,好问题
共 2 条评论48 - 夹心面包2019-02-131 关于insert造成死锁的情况,我之前做过测试,事务1并非只有insert,delete和update都可能造成死锁问题,核心还是插入唯一值冲突导致的.我们线上的处理办法是 1 去掉唯一值检测 2减少重复值的插入 3降低并发线程数量 2 关于数据拷贝大表我建议采用pt-archiver,这个工具能自动控制频率和速度,效果很不错,提议在低峰期进行数据操作展开
作者回复: 👍,这两点都是很有用的建议
共 2 条评论46 - Mr.Strive.Z.H.L2019-02-27老师您好: 关于文中的锁描述有所疑惑。 文中出现过 共享的next-key锁 和 排他的next-key锁。 我们知道next-key是由 gap lock 和 行锁组成的。 我一直以来的认知是 gap lock都是s锁,没有x锁。 而行锁有s锁和x锁。 比如 select………lock in share mode,行锁是s 锁。 比如select………for update,行锁就是x锁。 但是gap lock 始终是s锁。 文中直接描述next-key lock是排他的,总让我认为gap lock和行锁都是x锁。 不知道我理解得对不对?展开
作者回复: 是这样的,gap lock是无所谓S还是X的。 但是record lock 有。 Gap lock + 排他的record 就称作 排他的next-key lock 吧😄
共 2 条评论33 - 老杨同志2019-02-13课后问题: 我用的最多还是insert into select 。如果数量比较大,会加上limit 100,000这种。并且看看后面的select条件是否走索引。缺点是会锁select的表。方法二:导出成excel,然后拼sql 成 insert into values(),(),()的形式。方法3,写类似淘宝调动的定时任务,任务的逻辑是查询100条记录,然后多个线程分到几个任务执行,比如是个线程,每个线程10条记录,插入后,在查询新的100条记录处理。展开
作者回复: 👍
30 - Justin2019-02-15插入意向锁的gal lock和next key lock中的 gaplock互斥吗?
作者回复: 额, 这里我们要澄清一下哈 只有一个gap lock,就是 next key lock = gap lock + record lock; 我们说一个insert语句如果要插入一个间隙,而这个间隙上有gap lock的话,insert语句会被堵住,这个被堵住的效果,实现机制上是用插入意向锁和gap lock相互作用来实现的。 gap lock并不属于插入意向锁的一部分 ,就没有“插入意向锁的gal lock”这个概念哈
共 5 条评论24 - 一大只😴2019-02-13老师,我想问下:insert 语句出现唯一键冲突,会加next-key lock,而产生死锁的例子中,同样也是唯一键冲突却只加了记录锁,然后我按照唯一键冲突中的两个例子试了试 1、比如t表中有两条记录(19,19,19),(22,22,22),这时候我再insert (22,22,22)造成了主键冲突,这时候加的就是(19,22]的next-key lock,这个insert为啥不是等值查询? 2、根据死锁的例子,我又在t表中准备插入一行 session A :begin; insert into t values (25,25,25) session B :insert into t values (25,25,25) 这时候sessionB锁等待 session C:insert into t values (24,24,24) 锁等待,等B锁等待超时,session C插入成功 那这里的session B应该是加了个(22,25]的next-key lock,并没有因为是唯一键退化成记录锁 我想死锁的例子中t表已经有了(1,1,1),(2,2,2),(3,3,3),(4,4,4)4条记录,这时候insert (null,5,5),是不是加的(4,5]这个next-key lock,由于是整型并且间隙非常小,所以将他当成记录锁?展开
作者回复: “那这里的session B应该是加了个(22,25]的next-key lock,并没有因为是唯一键退化成记录锁” 由于insert主键冲突导致的锁,是不会退化的。 session B 加了next-key lock, 这样session C插入也要等待,然后等session B超时,释放了这个next-key lock,session C就可以执行了。 跟我们文中说的是一致的哦。 你这个验证挺合理的呀, 不会有因为“间隙非常小,所以将他当成记录锁”这种逻辑哈, a和a+1之间也是有间隙的😆。 不过这个是个好的实验和好问题👍
共 2 条评论18 - roaming2019-02-13MySQL8.0.12环境下, 执行insert into t(c,d) (select c+1, d from t force index(c) order by c desc limit 1); slow log Rows_examined: 2 Innodb_rows_read 的值增加1 是不是MySQL8进行了优化,先把子查询的结果读出来,再写入临时表?展开
作者回复: 看来是的了, 👍,很好的验证,我加到明天文章末尾说明
12 - inrtyx2019-04-07现在一般都用utf8mb4?
作者回复: 要看需求,不过因为表情用的很多了,utf8mb4很常用了
12 - 信信2019-02-17老师好,文中提到:insert into t2(c,d) (select c+1, d from t force index(c) order by c desc limit 1)的加锁范围是表 t 索引 c 上的 (4,supremum] 这个 next-key lock 和主键索引上 id=4 这一行。 可是如果我把表t的id为3这行先删除,再执行这个insert...select,那么别的会话执行insert into t values(3,3,3)会被阻塞,这说明4之前也是有间隙锁的? 另外,select c+1, d from t force index(c) order by c desc limit 1 for update 是不是不能用作等值查询那样分析?因为如果算等值查询,根据优化1是没有间隙锁的。展开
作者回复: 你说的对,这里其实是“向左扫描”,加锁范围应该是(3,4] 和 (4, supremum]。 👍
12 - Justin2019-02-15为什么insert 还会使用到next key lock 呢 ,我记得我原来看的资料写的是插入使用的是插入意向锁啊
作者回复: 是说插入碰到唯一键冲突的时候才会哈
10 - 信信2019-02-14老师好, 图6下方“发生主键冲突的时候”是不是应该改为“发生唯一键冲突的时候”?因为c不是主键。 还有,图7下方:T2时刻session b 发现“唯一键冲突”,这里为啥不是锁冲突?因为如果没有锁冲突,仅有唯一键冲突,就对应图6的情况,这时加的是next-key lock,而不仅仅是记录锁了。展开
作者回复: 1. 你说得对,👍细致,发起勘误了哈 2. 图7 这里说的唯一键冲突,就是发现“已经有一个c=5的行存在”,所以转为加next-key lock,没有单独的加行锁的逻辑哈
共 3 条评论9 - 王伯轩2019-02-19老师你好,去年双11碰到了dbcrash掉的情况.至今没有找到答案,心里渗得慌.老师帮忙分析下. 我是一个开发,关于db的知识更多是在应用和基本原理上面,实在是找不到原因. 我也搜了一些资料 感觉像是mysql的bug,不过在其buglist中没有找到完全一致的,当然也可能是我们业务也许导致库的压力大的原因. 应用端看到的现象是db没有响应,应用需要访问db的线程全部僵死.db表现是hang住 , 当时的诊断日志如下,表面表现为一直获取不到latch锁(被一个insert线程持有不释放) https://note.youdao.com/ynoteshare1/index.html?id=1771445db3ff1e08cbdd8328ea6765a7&type=note#/ 隔离级别是rr 同样的crash双11当天后面又出现了一次(哭死), 都是重启数据库解决的, 后面应用层面做了一样优化,没有再crash过,优化主要如下: 1.减小读压力,去除一些不必要的查询, 2.优化前,有并发事务写和查询同一条数据记录,即事务a执行insert 尚未提交,事务b就来查询(快照读),优化后保证查询时insert事务已经提交展开
作者回复: 这就是压力太大了。。 一般伴随着ioutil很大,语句执行特别慢,别的语句就被堵着等锁,等超时就自己crash
共 4 条评论8 - phpzheng2019-02-15循环插入数据,然后拿着刚刚插入的主键id,更新数据。请问怎么提高这个情况的效率
作者回复: insert以后 select last_insert_id, 再update, 只能这么做啦 如果要快一些,可能可以考虑减少交互,比如写成存储过程
7 - 常超2019-03-31对这个SQL 执行过程的解释, insert into t(c,d) (select c+1, d from t force index(c) order by c desc limit 1); >由于实现上这个语句没有在子查询中就直接使用 limit 1,从而导致了这个语句的执行需要遍历整个表 t。 没太看懂“由于”那句话,既然在子查询中有了limit 1,为什么不能只把最后一条记录插入临时表呢? 老师能在说明一下吗?展开共 1 条评论6
- 夹心面包2019-02-15我来补充应用表空间迁移的场景 1 冷数据表的复制和迁移 2 大表数据的恢复,线上DDL操作失误,需要恢复时,利用备份+binlog进行恢复后,表空间迁移进行导入 对于热表数据的复制建议还是采用pt-archiver慢慢搞
作者回复: 👍
6 - 发条橙子 。2019-02-23老师,年后过来狂补课程了哈哈 , 看到老师的bug留言已经被fix掉准备在最新版本发布了呢。 这里我有一个疑问, 我之前以为只有更新的时候才会加锁, 参考前面的文章,innodb要先扫描表中数据,被扫描到的行要加锁 。 或者我们执行 select 的时候手动加上 排他锁 或者 共享锁,也会锁住。 这里老师讲到如果索引唯一键冲突, innodb为了做处理加了 next_key lock(S) 这个可以理解。 insert .. select 也是因为有 select 索引会加锁 也可以理解 问题 : 图7那个死锁的案例, session A 的时候 只是执行了 insert 语句,执行 insert的时候也没有select之类的,为什么也会在索引c上加个锁, 是什么时候加的呢 ??? 是 insert 语句有索引的话都会给索引加锁么??展开
作者回复: 不是都会,是在要写入的时候,发现有主键冲突,才会加上这个next-key lock的锁
共 2 条评论5 - 滔滔2019-02-21老师,有个问题insert into … on duplicate key update语句在发生冲突的时候是先加next key读锁,然后在执行后面的update语句时再给冲突记录加上写锁,从而把之前加的next key读锁变成了写锁,是这样的吗?
作者回复: 不是,发现冲突直接加的就是写锁
5 - 王伯轩2019-02-18内存锁 大大计划讲下么,实际中碰到内存锁被持有后一直不释放导致db直接crash掉
作者回复: 这个系列里没讲到了 这种我碰到比较多的是io压力特别大,导致有的事务执行不下去,但是占着锁 然后其他事务就拿不到锁,有一个600计时,超过就crash了
5