带你从根本理解mysql innodb的各种锁

2024-08-29 08:38

本文主要是介绍带你从根本理解mysql innodb的各种锁,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

锁的相关概念:

锁:数据库是一个多用户使用的共享资源。当多个用户并发地存取数据时,在数据库中就会产生多个事务同时存取同一数据的情况。若对并发操作不加控制就可能会读取和存储不正确的数据,破坏数据库的一致性。而锁就是并发控制的一个重要的手段。

表锁:对某个表进行加锁。能够保证不会出现死锁,但是发生锁冲突几率高,并发性能差。

排他锁(X):排他锁也叫“写锁”,获得某个表的排他锁的事务便拥有了对这个表的写权限,同时排他锁会阻止其他事务取得该表的共享锁和排他锁。

共享锁(S):共享锁也叫“读锁”,获得某个表的共享锁的事务便拥有对这个表的读权限,共享锁不会阻止其他事务获取该表的共享锁,但会阻止其他事务获取该表的排他锁。

意向排他锁(IX):表示事务打算给数据行加行排他锁。事务在给一个数据行加排他锁前必须先取得该表的IX锁。

意向共享锁(IS):表示事务打算给数据行加行共享锁。事务在给一个数据行加共享锁前必须先取得该表的IS锁。

行锁:对某行或某几行数据进行加锁。这样做可能出现死锁,但是发生锁冲突几率低,并发性能好。InnoDB行锁是通过给索引上的索引项加锁来实现的,如果没有索引的话不存在行锁。

排他锁(X):排他锁也叫“写锁”,获得数据行排他锁的事务便拥有了对所锁定数据的写权限,同时排他锁会阻止其他事务取得相同数据的共享锁和排他锁。

共享锁(S):共享锁也叫“读锁”,获得数据行共享锁的事务便拥有对所锁定数据的读权限,共享锁不会阻止其他事务获取锁定的数据的共享锁,但会阻止其他事务获取数据的排他锁。

间隙锁(gap):锁定索引上的一个范围上所有的数据(所有值在这个范围上的所有数据,包括表中不存在的数据),但不会锁住这个范围的边界值,其数学表达式为(a,b)。

Next-Key Lock(Record+Gap):锁定索引上的一个范围,并且锁定记录本身,相当于间隙锁+行锁,其数学表达式为[a,b]。

插入意向锁:事务要插入一条数据前需要申请的一种特殊的间隙锁,当要锁住的间隙中存在间隙锁(gap锁)时,出现阻塞,但是当该间隙存在其他插入意向锁且该插入意向锁所要插入的数据跟当前插入意向锁计划要插入的数据的索引值不同时并不阻塞。

特殊锁的引入缘由:

意向锁:在MySQL数据库中,表锁与表关联,行锁与索引关联,当我们想要获取到一个表中是否存在行锁需要遍历整柜表的索引信息,当表中的数据很多时会消耗大量的性能,索引innodb引擎并没有采取这种方式来使表锁和行锁间进行互相感应。但是如果表锁在加锁的时候不能知道当前表是否加了行锁可能会出现表锁和行锁出现冲突的情况,例如:

session A对表t中的某些行加了排他锁,之后session B对表t加排他锁,由于表t上没有加表锁,所以session B也能加锁成功。这时对于session A来说,t1中被加了排他锁的几行数据自己应该有修改的权限,而且其他会话不能对这几行数据进行修改,对于session B来说t1中的所有数据自己都拥有修改权限,而且其他会话不能对表中的任何数据进行修改,显然这两个session间的锁出现了冲突。

为了避免这种冲突,innodb引擎引入了意向锁的概念来对表中是否加了行锁进行标志。在加行锁前引擎会先对对应的表加一个意向锁,这时如果表上已经存在与之冲突的表锁,那么会话进入等待,如果没有出现冲突,则加锁成功。加锁成功后,如果其他会话要对表进行加锁,那么就会知道该表已经有某些行加了行锁,并根据要加的表锁和已加的意向锁是否冲突来决定加锁是否成功。

间隙锁:我们知道行锁是在索引的基础上实现的,在可重复读(RR)隔离级别下,为了防止在同一事物的两次通过条件读取数据的过程中有其他事务对符合条件的这段数据进行修改从而导致读到的结果不一致的情况,innodb引擎在RR隔离级别下引入了间隙锁来锁住两个数据间的间隙(不包括两个数据本身,但是两个数据间即使不存在的数据也会被加锁),从而防止在某个会话执行过程中有其他事务对这个范围的数据进行修改导致前后两次读取数据库读到的数据不一致,即不可重复读的情况。

ps.在innodb引擎中普通的select语句(eg. select * from tab where id >3 and id<10)会获取当前时刻表的一个快照,所以并没有对这块数据加间隙锁,只有在使用类似select * from tab where id >3 and id<10 for update的形式的select语句才会对数据加间隙锁。

插入意向锁:间隙锁的目的是为了防止其他会话对锁定范围内的数据进行增删改的操作。当有一个会话对某个范围加了一个间隙锁,这时如果有另一个会话要对间隙锁锁定范围内的某行数据进行删改操作,需要先获取修改该行的排他锁,但是间隙锁与排他锁冲突,所以该修改就会被阻塞。但是,如果另一个会话如果要插入一个锁定范围内的数据,由于插入前这行数据不存在,所以没有办法加排他锁,所以该修改不会被阻塞。为了让间隙锁能够阻止其他会话对锁定范围内的数据进行插入操作,innodb引入了插入意向锁的概念,当一个会话要插入一条数据时,会先获取一个插入意向锁,如果该数据所在位置处于某个间隙锁的范围内时,无法成功获取到所需的插入意向锁,该插入操作被阻塞。

锁的兼容性关系

表锁:

行锁:

这篇关于带你从根本理解mysql innodb的各种锁的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SQL中的外键约束

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

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

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

如何去写一手好SQL

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

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

性能分析之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_为前缀并以日期为后缀,备份

【生成模型系列(初级)】嵌入(Embedding)方程——自然语言处理的数学灵魂【通俗理解】

【通俗理解】嵌入(Embedding)方程——自然语言处理的数学灵魂 关键词提炼 #嵌入方程 #自然语言处理 #词向量 #机器学习 #神经网络 #向量空间模型 #Siri #Google翻译 #AlexNet 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言

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

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

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝