mysql锁机制要览+示例讲解

2023-11-23 17:10

本文主要是介绍mysql锁机制要览+示例讲解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

作者:杨恒

背景

2、隔离级别

理论

1、read uncommited
可以读取未提交记录。此隔离级别,不会使用,忽略
2、read commited
针对当前读,rc隔离级别保证对读取到的记录加锁(记录锁),存在幻读现象
3、repeatable read
针对当前读,rr隔离级别保证对读取到的记录加锁(记录锁),同时保证对读取的范围加锁,新的满足查询条件的记录不能插入(间隙锁),不存在幻读现象
4、seriablizable
从mvcc并发控制退化为基于锁的并发控制。不区别快照读和当前读,所有的读操作均为当前读,读加读锁(S锁),写加写锁(X锁)

隔离级别脏读可能性不可重复读可能性幻读可能性加锁读
read uncommitedyesyesyesno
read commitednoyesyesno
repeatable readnonoyesno
serializablenononoyes

5、rc 与rr对比:
set global transaction isolation level read commited;
set global transaction isolation level repeatable read;

测试

drop table if exists t;
create table t(id int,name varchar(10),key idx_id(id),primary key(name))engine=innodb;
insert into t values(1,‘a’),(3,‘c’),(5,‘e’),(8,‘g’),(11,‘j’);

t1t2
begin;
select * from t where id=1;
commit;
begin;
update t set name=‘yy’ where id=1;
commit;

3、锁

理论

基本锁:共享锁与排它锁

mysql允许拿到S锁的事务读一行,允许拿到X锁的事务更新或删除一行
加了S锁的记录,允许其他事务再加S锁,不允许其他事务再加X锁;
加了X锁的记录,不允许其他事务再加S锁或X锁

mysql对外提供加这两种锁的语法如下:
加S锁: select … lock in share mode
加X锁: select … for update

意向锁(表级锁):意向共享锁(IS锁)和意向排它锁(IX锁)

事务在请求S锁和X锁前,需要先获得对应的IS、IX锁
意向锁产生的主要目的是为了处理行锁和表锁之间的冲突,用于表明“某个事务正在某一行上持有了锁,或者准备去持有锁”

共享锁、排它锁与意向锁的兼容矩阵(先行后列)

 XIXSIS
X冲突冲突冲突冲突
IX冲突冲突兼容兼容
S冲突冲突兼容兼容
IS冲突兼容兼容兼容
行锁
记录锁

仅仅锁住索引记录的一行。单行索引记录上加锁,record lock锁住的永远是索引,而非记录本身,即使该表上没有任何索引,那么innodb会在后台创建一个隐藏的聚簇主键索引,那么锁住的就是这个隐藏的聚簇主键索引。
所以说当一条sql没有走任何索引时,那么将会在每一条聚簇索引后面加X锁,这个类似于表锁,但原理上和表锁应该是完全不同的。(Is it true??)

间隙锁

区间锁,仅仅锁住一个索引区间(开区间)
在索引记录之间的间隙中加锁,或者是在某一条索引记录之前或之后加锁,并不包括该索引记录本身

next-key锁

record lock + gap lock,左开右闭区间。默认情况下,innodb使用next-key locks来锁定记录。但当查询的索引含有唯一属性的时候,next-key lock会进行优化,将其降级为record lock,即仅锁住索引本身,不是范围

插入意向锁:特殊的间隙锁

gap lock中存在一种插入意向锁,在insert操作时产生。在多事务同时写入不同数据至同一索引间隙的时候,并不需要等待其他事务完成,不会发生锁等待。
假设有一个记录索引包含键值4和7,不同的事务分别插入5和6,每个事务都会产生一个加在4-7之间的插入意向锁,获取在插入行上的排它锁,但是不会被互相锁住,因为数据行并不冲突。

行锁的兼容矩阵 (先列(如gap)后行的锁定顺序)

 gapinsertrecordnext-key
gap兼容兼容兼容兼容
insert冲突兼容兼容冲突
record兼容兼容冲突冲突
next-key兼容兼容冲突冲突

1、已有的insert锁不阻止任何准备加的锁
2、gap、next-key会阻止insert
3、gap和record、next-key不会冲突
4、record和record、next-key之间相互冲突

4、测试

实例一、

t1t2
begin;
select * from t where id=8 for update;
commit;
begin;
insert into t values(4);
insert into t values(5);
insert into t values(6);
insert into t values(11);
insert into t values(12);
rollback;

drop table if exists t;
create table t(id int,key idx_a(id))engine=innodb;
insert into t values(1),(3),(5),(8),(11);
select * from t;

分析:
因为innoDB对于行的查询都是采用了next-key lock的算法,锁定的不是单个值,而是一个范围。上面索引值有(1,3,5,8,11),其记录的gap区间如下:是一个左开右闭的空间(原因是默认主键的有序自增的特性,结合后面的例子说明)(-∞,1],(1,3],(3,5],(5,8],(8,11],(11,+∞)
innoDB存储引擎还会对辅助索引下一个键值加上gap lock。

实例二、

t1t2
begin;
delete from t where id=8;
commit;
begin;
insert into t(id,name) values(6,‘f’);
insert into t(id,name) values(5,‘e1’);
insert into t(id,name) values(8,‘gg’);
insert into t(id,name) values(10,‘p’);
insert into t(id,name) values(11,‘iz’);
insert into t(id,name) values(5,‘cz’);

分析:因为会话1已经对id=8的记录加了一个X锁,由于是RR隔离级别,innodb要防止幻读需要加gap锁:即id=5(8的左边),id=11(8的右边)之间需要加间隙锁(gap)。这样[5,e]和[8,g],[8,g]和[11,j]之间的数据都要被锁。上面测试已经验证了这一点,根据索引的有序性,数据按照主键name排序,后面写入的[5,cz] ([5,e]的左边)和[11,ja] ([11,j]的右边)不属于上面的范围从而可以写入。

实例三、

t1t2
begin;
select * from t where id=8 for update;
commit;
begin;
insert into t values(7);
insert into t values(9);
rollback;

drop table if exists t;
create table t(id int primary key)engine=innodb;
insert into t values(1),(3),(5),(8),(11);
select * from t;

分析:
因为innoDB对于行的查询都是采用了next-key lock的算法,锁定的不是单个值,而是一个范围,按照这个方法和第一个测试结果一样。但是,当查询的索引含有唯一属性的时候,next-key lock会进行优化,将其降级为record lock,即仅锁住索引本身,不是范围。

实例四、

t1t2
begin;
select * from t where id=15 for update;
commit;
begin;
insert into t(id,name) values(10,‘k’);
insert into t(id,name) values(12,‘k’);
rollback;

drop table if exists t;
create table t(id int,name varchar(10),primary key(id))engine=innodb;
insert into t values(1,‘a’),(3,‘c’),(5,‘e’),(8,‘g’),(11,‘j’);
select * from t;

分析:
通过主键或者唯一索引来锁定不存在的值,也会产生gap锁定。

5、死锁

show engine innodb status;

duplicate key error 引发的死锁

drop table if exists t;
create table t(id int(10) unsigned not null ,
name varchar(20) not null default ‘’,
age int(11) not null default ‘0’ ,
stage int(11) not null default ‘0’ ,
primary key(id),
unique key udx_name(name),
key idx_stage(stage))engine=innodb;

insert into t(id,name,age,stage) values(1,‘yst’,11,8),(2,‘dxj’,7,4),(3,‘lb’,13,7),(4,‘zsq’,5,7),(5,‘lxr’,13,4);
select * from t;
select * from information_schema.innodb_locks;

t1t2t3
begin;
insert into t values(6,‘test’,12,3);
rollback;
begin;
insert into t values(6,‘test’,12,3);
OK
begin;
insert into t values(6,‘test’,12,3);
ERROR

死锁成因
事务t1成功插入记录,并获得索引id=6上的排他记录锁(lock_x)
紧接着事务t2、t3也开始插入记录,请求排他插入意向锁(lock_insert_intention);但由于发生重复唯一键冲突,各自请求的排他记录锁(lock_x)转成共享记录锁(lock_s)

t1回滚释放索引id=6上的排他记录锁(lock_x),t2和t3都要请求索引id=6上的排他记录锁(lock_x)。
由于x锁和s锁互斥,t2和t3都等待对方释放s锁。
于是,死锁便产生了。

如果此场景下,只有两个事务t1与t2或者t1与t3,则不会引发如上死锁情况发生。

gap与insert intention冲突引发的死锁

drop table if exists t;
create table t(a int(11) not null, b int(11) default null,primary key(a),key idx_b(b))engine=innodb default charset=utf8;
insert into t(a,b) values(1,2),(2,3),(3,4),(11,55);
select * from t;

t1t2
begin;
select * from t where b=6 for update;
insert into t values(4,5);
commit;
begin;
select * from t where b=8 for update;
insert into t values(4,5);
commit;

死锁成因
事务t1执行查询语句,在索引b=6上加排他next-key锁(lock_x),会锁住idx_b索引范围(4,22)。
事务t2执行查询语句,在索引b=8上加排他next-key锁(lock_x),会锁住idx_b索引范围(4,22)。
由于请求的gap与已持有的gap是兼容的,因此,事务t2在idx_b索引范围(4,22)也能加锁成功。

事务t1执行插入语句,会先加他insert intention锁。由于请求的insert intention锁与已有的gap锁不兼容,则事务t1等待t2释放gap锁。
事务t2执行插入语句,也会等待t1释放gap锁。于是,死锁便产生了。

6、程序应用

这里写图片描述
这里写图片描述

这篇关于mysql锁机制要览+示例讲解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/419441

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

SQL中的外键约束

外键约束用于表示两张表中的指标连接关系。外键约束的作用主要有以下三点: 1.确保子表中的某个字段(外键)只能引用父表中的有效记录2.主表中的列被删除时,子表中的关联列也会被删除3.主表中的列更新时,子表中的关联元素也会被更新 子表中的元素指向主表 以下是一个外键约束的实例展示

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

如何去写一手好SQL

MySQL性能 最大数据量 抛开数据量和并发数,谈性能都是耍流氓。MySQL没有限制单表最大记录数,它取决于操作系统对文件大小的限制。 《阿里巴巴Java开发手册》提出单表行数超过500万行或者单表容量超过2GB,才推荐分库分表。性能由综合因素决定,抛开业务复杂度,影响程度依次是硬件配置、MySQL配置、数据表设计、索引优化。500万这个值仅供参考,并非铁律。 博主曾经操作过超过4亿行数据

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

MySQL数据库宕机,启动不起来,教你一招搞定!

作者介绍:老苏,10余年DBA工作运维经验,擅长Oracle、MySQL、PG、Mongodb数据库运维(如安装迁移,性能优化、故障应急处理等)公众号:老苏畅谈运维欢迎关注本人公众号,更多精彩与您分享。 MySQL数据库宕机,数据页损坏问题,启动不起来,该如何排查和解决,本文将为你说明具体的排查过程。 查看MySQL error日志 查看 MySQL error日志,排查哪个表(表空间

MySQL高性能优化规范

前言:      笔者最近上班途中突然想丰富下自己的数据库优化技能。于是在查阅了多篇文章后,总结出了这篇! 数据库命令规范 所有数据库对象名称必须使用小写字母并用下划线分割 所有数据库对象名称禁止使用mysql保留关键字(如果表名中包含关键字查询时,需要将其用单引号括起来) 数据库对象的命名要能做到见名识意,并且最后不要超过32个字符 临时库表必须以tmp_为前缀并以日期为后缀,备份

[MySQL表的增删改查-进阶]

🌈个人主页:努力学编程’ ⛅个人推荐: c语言从初阶到进阶 JavaEE详解 数据结构 ⚡学好数据结构,刷题刻不容缓:点击一起刷题 🌙心灵鸡汤:总有人要赢,为什么不能是我呢 💻💻💻数据库约束 🔭🔭🔭约束类型 not null: 指示某列不能存储 NULL 值unique: 保证某列的每行必须有唯一的值default: 规定没有给列赋值时的默认值.primary key:

Java ArrayList扩容机制 (源码解读)

结论:初始长度为10,若所需长度小于1.5倍原长度,则按照1.5倍扩容。若不够用则按照所需长度扩容。 一. 明确类内部重要变量含义         1:数组默认长度         2:这是一个共享的空数组实例,用于明确创建长度为0时的ArrayList ,比如通过 new ArrayList<>(0),ArrayList 内部的数组 elementData 会指向这个 EMPTY_EL

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能