MySQL InnoDB Lock Modes

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 行锁的S锁和X锁 Innodb实施标准的行级锁,其中有两种类型的锁:Shared lock即共享锁,S锁。如果事务对数据行r持有S锁,那么允许其它事务对数据行r持有S锁,但不允许其它事务对数据行持有X锁。

行锁的S锁和X锁

Innodb实施标准的行级锁,其中有两种类型的锁:
Shared lock即共享锁,S锁。如果事务对数据行r持有S锁,那么允许其它事务对数据行r持有S锁,但不允许其它事务对数据行持有X锁。
Exclusive lock即排它锁,X锁。如果事务T1对数据行r持有X锁,那么就不允许其它事务对数据行持有S锁或X锁,除非等到T1释放r上的X锁

意向锁(Intention lock)

Innodb支持多粒度锁,即允许记录锁(record locks)和表锁共存(InnoDB supports multiple granularity locking which permits coexistence of record locks and locks on entire tables)。为了在多个粒度级别上实现锁定,innodb引进了意向锁。有两种意向锁:
Intention shared(IS):意向共享锁,表示事务T将要对表中数据行加S锁,而先在表级别上加的就是IS锁;
Intention exclusive(IX):意向排它锁,表示事务T将要对表中数据行加X锁,而先在表级别上加的就是IX锁;

意图锁是表锁,它指示事务稍后将在表数据行上加何种类型的锁。IS锁表示事务稍后会在数据行上加S锁,IX锁表示事务稍后会在数据行上加X锁。常见的,像“select … lock in share mode”语句会加IS锁,“select … for update”语句会加IX锁。

所以,引入意图锁的目的就是为了支持多粒度锁定,并能通过意图锁显示事务锁定了表中的某些行,或者将要锁定表中的某些行。

意图锁的协议如下:
一个事务要想获得行上的S锁,必须先获取该表上的IS或者更强的锁;
一个事务要想获得行上的X锁,必须先获取该表上的IX锁。

四种锁之间的兼容性如下:
_1

如果事务请求的锁与现有锁兼容,则授予该事务锁;但如果与现有锁冲突,则不授予该事务锁,事务需要等待现有锁被释放后才能获取锁。

说明:
个人认为,上面的S和X锁是表级别的S和X锁。但之前和朋友讨论,说没有表级别的S和X锁,只有行级别的S和X锁,并上面指的是行级别的S和X锁。后来发现在姜承尧的innodb内幕这本书中,也提到上面指的是行级别的S和X锁。
因此有些疑惑的就是:锁的粒度不同,如何讨论兼容性?更何况意图锁引入的目的就是为了支持mysql的多粒度锁定。

死锁

死锁演示:
会话1上开启事务T1:

mysql> start transaction;
mysql> select * from ecs_payment where pay_id=2332 lock in share mode;
+--------+----------+----------+---------+----------+-----------+------------+---------+--------+-----------+
| pay_id | pay_code | pay_name | pay_fee | pay_desc | pay_order | pay_config | enabled | is_cod | is_online |
+--------+----------+----------+---------+----------+-----------+------------+---------+--------+-----------+
|   2332 | 83       | ^{}      | 4090.99 | bwlw]^k  |        13 | lb^wkw]    |       1 |      0 |         0 |
+--------+----------+----------+---------+----------+-----------+------------+---------+--------+-----------+

会话2上开启事务T2:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> update ecs_payment set pay_name='tiatiao' where pay_id=2332;    --被卡住

会话1的事务T1再执行操作:

mysql> delete from ecs_payment where pay_id=2332;
Query OK, 1 row affected (0.00 sec)

这时再看会话2的事务T2,会输出如下信息:

ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

发生死锁的四个必要条件:
互斥条件
不可剥夺
请求与保持
循环等待

注意:
如果InnoDB监视器输出的最新检测到的死锁部分包含如下输出信息:

TOO DEEP OR LONG SEARCH IN THE LOCK TABLE WAITS-FOR GRAPH, WE WILL ROLL BACK FOLLOWING TRANSACTION

这表明锁等待列表上的事务数量已经达到了200个的限制,超过200个事务的等待列表被视为死锁,这个数量限制和参数LOCK_MAX_DEPTH_IN_DEADLOCK_CHECK有关

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
27天前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的数据文件与重做日志文件
本文介绍了MySQL InnoDB存储引擎中的数据文件和重做日志文件。数据文件包括`.ibd`和`ibdata`文件,用于存放InnoDB数据和索引。重做日志文件(redo log)确保数据的可靠性和事务的持久性,其大小和路径可由相关参数配置。文章还提供了视频讲解和示例代码。
131 11
【赵渝强老师】MySQL InnoDB的数据文件与重做日志文件
|
7天前
|
存储 关系型数据库 MySQL
MySQL引擎InnoDB和MyISAM的区别?
InnoDB是MySQL默认的事务型存储引擎,支持事务、行级锁、MVCC、在线热备份等特性,主索引为聚簇索引,适用于高并发、高可靠性的场景。MyISAM设计简单,支持压缩表、空间索引,但不支持事务和行级锁,适合读多写少、不要求事务的场景。
30 9
|
27天前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的表空间
InnoDB是MySQL默认的存储引擎,主要由存储结构、内存结构和线程结构组成。其存储结构分为逻辑和物理两部分,逻辑存储结构包括表空间、段、区和页。表空间是InnoDB逻辑结构的最高层,所有数据都存放在其中。默认情况下,InnoDB有一个共享表空间ibdata1,用于存放撤销信息、系统事务信息等。启用参数`innodb_file_per_table`后,每张表的数据可以单独存放在一个表空间内,但撤销信息等仍存放在共享表空间中。
|
27天前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的段、区和页
MySQL的InnoDB存储引擎逻辑存储结构与Oracle相似,包括表空间、段、区和页。表空间由段和页组成,段包括数据段、索引段等。区是1MB的连续空间,页是16KB的最小物理存储单位。InnoDB是面向行的存储引擎,每个页最多可存放7992行记录。
|
27天前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL的InnoDB存储引擎
InnoDB是MySQL的默认存储引擎,广泛应用于互联网公司。它支持事务、行级锁、外键和高效处理大量数据。InnoDB的主要特性包括解决不可重复读和幻读问题、高并发度、B+树索引等。其存储结构分为逻辑和物理两部分,内存结构类似Oracle的SGA和PGA,线程结构包括主线程、I/O线程和其他辅助线程。
【赵渝强老师】MySQL的InnoDB存储引擎
|
2月前
|
关系型数据库 MySQL Java
MySQL数据锁:Record Lock,Gap Lock 和 Next-Key Lock
本文基于 MySQL 8.0.30 版本及 InnoDB 引擎,深入解析三种行锁机制:记录锁(Record Lock)、间隙锁(Gap Lock)和临键锁(Next-key Lock)。记录锁锁定索引记录,确保事务唯一修改;间隙锁锁定索引间的间隙,防止新记录插入;临键锁结合两者,锁定范围并记录自身,有效避免幻读现象。通过具体示例展示了不同锁的作用机制及其在并发控制中的应用。
191 2
|
7天前
|
SQL 关系型数据库 MySQL
MySQL导入.sql文件后数据库乱码问题
本文分析了导入.sql文件后数据库备注出现乱码的原因,包括字符集不匹配、备注内容编码问题及MySQL版本或配置问题,并提供了详细的解决步骤,如检查和统一字符集设置、修改客户端连接方式、检查MySQL配置等,确保导入过程顺利。
|
26天前
|
SQL 关系型数据库 MySQL
12 PHP配置数据库MySQL
路老师分享了PHP操作MySQL数据库的方法,包括安装并连接MySQL服务器、选择数据库、执行SQL语句(如插入、更新、删除和查询),以及将结果集返回到数组。通过具体示例代码,详细介绍了每一步的操作流程,帮助读者快速入门PHP与MySQL的交互。
34 1
|
29天前
|
SQL 关系型数据库 MySQL
go语言数据库中mysql驱动安装
【11月更文挑战第2天】
39 4
|
1月前
|
监控 关系型数据库 MySQL
数据库优化:MySQL索引策略与查询性能调优实战
【10月更文挑战第27天】本文深入探讨了MySQL的索引策略和查询性能调优技巧。通过介绍B-Tree索引、哈希索引和全文索引等不同类型,以及如何创建和维护索引,结合实战案例分析查询执行计划,帮助读者掌握提升查询性能的方法。定期优化索引和调整查询语句是提高数据库性能的关键。
193 1