MySQL InnoDB Lock Modes

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 行锁的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
目录
相关文章
|
9天前
|
存储 关系型数据库 MySQL
Mysql索引:深入理解InnoDb聚集索引与MyisAm非聚集索引
通过本文的介绍,希望您能深入理解InnoDB聚集索引与MyISAM非聚集索引的概念、结构和应用场景,从而在实际工作中灵活运用这些知识,优化数据库性能。
57 7
|
1月前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的数据文件与重做日志文件
本文介绍了MySQL InnoDB存储引擎中的数据文件和重做日志文件。数据文件包括`.ibd`和`ibdata`文件,用于存放InnoDB数据和索引。重做日志文件(redo log)确保数据的可靠性和事务的持久性,其大小和路径可由相关参数配置。文章还提供了视频讲解和示例代码。
141 11
【赵渝强老师】MySQL InnoDB的数据文件与重做日志文件
|
16天前
|
存储 关系型数据库 MySQL
MySQL引擎InnoDB和MyISAM的区别?
InnoDB是MySQL默认的事务型存储引擎,支持事务、行级锁、MVCC、在线热备份等特性,主索引为聚簇索引,适用于高并发、高可靠性的场景。MyISAM设计简单,支持压缩表、空间索引,但不支持事务和行级锁,适合读多写少、不要求事务的场景。
48 9
|
1月前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的表空间
InnoDB是MySQL默认的存储引擎,主要由存储结构、内存结构和线程结构组成。其存储结构分为逻辑和物理两部分,逻辑存储结构包括表空间、段、区和页。表空间是InnoDB逻辑结构的最高层,所有数据都存放在其中。默认情况下,InnoDB有一个共享表空间ibdata1,用于存放撤销信息、系统事务信息等。启用参数`innodb_file_per_table`后,每张表的数据可以单独存放在一个表空间内,但撤销信息等仍存放在共享表空间中。
|
1月前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL InnoDB的段、区和页
MySQL的InnoDB存储引擎逻辑存储结构与Oracle相似,包括表空间、段、区和页。表空间由段和页组成,段包括数据段、索引段等。区是1MB的连续空间,页是16KB的最小物理存储单位。InnoDB是面向行的存储引擎,每个页最多可存放7992行记录。
|
1月前
|
存储 Oracle 关系型数据库
【赵渝强老师】MySQL的InnoDB存储引擎
InnoDB是MySQL的默认存储引擎,广泛应用于互联网公司。它支持事务、行级锁、外键和高效处理大量数据。InnoDB的主要特性包括解决不可重复读和幻读问题、高并发度、B+树索引等。其存储结构分为逻辑和物理两部分,内存结构类似Oracle的SGA和PGA,线程结构包括主线程、I/O线程和其他辅助线程。
【赵渝强老师】MySQL的InnoDB存储引擎
|
6月前
|
存储 关系型数据库 MySQL
MySQL数据库进阶第六篇(InnoDB引擎架构,事务原理,MVCC)
MySQL数据库进阶第六篇(InnoDB引擎架构,事务原理,MVCC)
|
2月前
|
存储 缓存 关系型数据库
详细解析MySQL中的innodb和myisam
总之,InnoDB和MyISAM各有千秋,选择合适的存储引擎应基于对应用程序特性的深入理解,以及对性能、数据完整性和可扩展性的综合考量。随着技术发展,InnoDB因其全面的功能和日益优化的性能,逐渐成为更广泛场景下的首选。然而,在特定条件下,MyISAM依然保留其独特的价值。
167 0
|
4月前
|
监控 关系型数据库 MySQL
在Linux中,mysql的innodb如何定位锁问题?
在Linux中,mysql的innodb如何定位锁问题?
|
4月前
|
SQL 存储 关系型数据库
"MySQL增列必锁表?揭秘InnoDB在线DDL,让你的数据库操作飞一般,性能无忧!"
【8月更文挑战第11天】在数据库领域,MySQL凭借其稳定高效的表现深受开发者喜爱。对于是否会在给数据表添加列时锁表的问题,MySQL的行为受版本、存储引擎等因素影响。从5.6版起,InnoDB支持在线DDL,可在改动表结构时保持表的可访问性,避免长时间锁表。而MyISAM等则需锁表完成操作。例如,在使用InnoDB的表上运行`ALTER TABLE users ADD COLUMN email VARCHAR(255);`时,通常不会完全锁表。虽然在线DDL提高了灵活性,但复杂操作或大表变更仍可能暂时影响性能。因此,进行结构变更前应评估其影响并择机执行。
76 6