面试让 HR 都能听懂的 MySQL 锁机制详解
linux 服务器开发相关视频解析:
腾讯、阿里等大厂面试,不了解这些MySQL, InnoDB技术,何以征服面试官
c/c++ linux 服务器开发免费学习地址:c/c++ linux后台服务器高级架构师
腾讯云数据库负责人说过:“我们面试 MySQL 同事时只考察两点,索引和锁”。言简意赅,MySQL 锁的重要性不言而喻。
本文通过一场“面试“,带你通俗易懂的掌握 MySQL 各种锁机制,希望可以帮到你!
XX:面试官好!
面试官:你好,小 XX 啊,看你简历写着精通 MySQL 锁,你认为精通应该是啥水平呢?
XX:emmm…我是个老实人啊,我认为精通就是,比面试官知道的多就完了。。
面试官:??怎么有种似曾相识的感觉?
面试官:行,那你先给我说说 MySQL 设计这个锁是干啥用的呀?
XX:数据库锁设计的初衷是处理并发问题。作为多用户共享的资源,当出现并发访问的时候,为了保证数据的一致性,数据库需要合理地控制资源的访问规则。而锁就是用来实现这些访问规则的重要机制。
简单说,数据表就像公共厕所。emmm…换个下饭的说法,数据表就好比您开的一家酒店,而每行数据就像酒店的房间,如果大家随意进出,就会出现多人抢夺同一个房间的情况,而在房间上装上锁,申请到钥匙的人才可以入住并且将房间锁起来,其他人只有等他用完退房后才可以再次使用,这样保证了房间的一致性,方便酒店进行管理。
MySQL 锁机制的初衷便是如此,当然,MySQL 数据库由于其自身架构的特点,存在多种数据存储引擎,每种存储引擎所针对的应用场景特点都不太一样,为了满足各自特定应用场景的需求,每种存储引擎的锁定机制都是为各自所面对的特定场景而优化设计,所以各存储引擎的锁定机制也有较大区别。
面试官:嗯,那你说一下 MySQL 都分为哪些锁。
XX:
按锁粒度从大到小分类:表锁,页锁和行锁;以及特殊场景下使用的全局锁
如果按锁级别分类则有:共享(读)锁、排他(写)锁、意向共享(读)锁、意向排他(写)锁;
以及 Innodb 引擎为解决幻读等并发场景下事务存在的数据问题,引入的 Record Lock(行记录锁)、Gap Lock(间隙锁)、Next-key Lock(Record Lock + Gap Lock 结合)等;
还有就是我们面向编程的两种锁思想:悲观锁、乐观锁。
面试官:元芳(HR 小姐姐)你怎么看?
HR 小姐姐:。。。
面试官:小 XX 啊,那你来谈一谈你对表锁、行锁的理解吧。
表锁
XX:表级别的锁定是 MySQL 各存储引擎中最大颗粒度的锁定机制。该锁定机制最大的特点是实现逻辑非常简单,带来的系统负面影响最小。所以获取锁和释放锁的速度很快。由于表级锁一次会将整个表锁定,所以可以很好的避免困扰我们的死锁问题。
当然,锁定颗粒度大所带来最大的负面影响就是出现锁定资源争用的概率也会最高,大大降低并发度。
使用表级锁定的主要是 MyISAM,MEMORY,CSV 等一些非事务性存储引擎。
行锁
XX:与表锁正相反,行锁最大的特点就是锁定对象的颗粒度很小,也是目前各大数据库管理软件所实现的锁定颗粒度最小的。由于锁定颗粒度很小,所以发生锁定资源争用的概率也最小,能够给予应用程序尽可能大的并发处理能力从而提高系统的整体性能。
虽然能够在并发处理能力上面有较大的优势,但是行级锁定也因此带来了不少弊端。由于锁定资源的颗粒度很小,所以每次获取锁和释放锁需要做的事情也更多,带来的消耗自然也就更大了。此外,行级锁定也最容易发生死锁。
使用行级锁定的主要是 InnoDB 存储引擎。
适用场景:从锁的角度来说,表级锁更适合于以查询为主,只有少量按索引条件更新数据的应用,如 Web 应用;而行级锁则更适合于有大量按索引条件并发更新数据的情况,同时又有并发查询的应用场景。
页锁
除了表锁、行锁外,MySQL 还有一种相对偏中性的页级锁,页锁是 MySQL 中比较独特的一种锁定级别,在其他数据库管理软件中也并不是太常见。页级锁定的特点是锁定颗粒度介于行级锁定与表级锁之间,所以获取锁定所需要的资源开销,以及所能提供的并发处理能力也同样是介于上面二者之间。另外,页级锁定和行级锁定一样,会发生死锁。
使用页级锁定的主要是 BerkeleyDB 存储引擎。
面试官:那全局锁是什么时候用的呢?
【文章福利】需要 C/C++ Linux 服务器架构师学习资料加群 812855908(资料包括 C/C++,Linux,golang 技术,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK,ffmpeg 等)
全局锁
XX:首先全局锁,是对整个数据库实例加锁。使用场景一般在全库逻辑备份时。
MySQL 提供加全局读锁的命令:Flush tables with read lock (FTWRL)
这个命令可以使整个库处于只读状态。使用该命令之后,数据更新语句、数据定义语句和更新类事务的提交语句等修改数据库的操作都会被阻塞。
风险:
如果在主库备份,在备份期间不能更新,业务停摆
如果在从库备份,备份期间不能执行主库同步的 binlog,导致主从延迟同步
还有一种锁全局的方式:set global readonly=true ,相当于将整个库设置成只读状态,但这种修改 global 配置量级较重,和全局锁不同的是:如果执行 Flush tables with read lock 命令后,如果客户端发生异常断开,那么 MySQL 会自动释放这个全局锁,整个库回到可以正常更新的状态。但将库设置为 readonly 后,客户端发生异常断开,数据库依旧会保持 readonly 状态,会导致整个库长时间处于不可写状态,试想一下微信只能看,不能打字~~
HR 小姐姐:那微信不就完蛋了?
XX:是啊,抓紧找老实人背锅!
面试官:不错,你把这几种锁的侧重点都表述清楚了。那你再说一下你对不同级别的那几种锁的使用场景和理解吧?
XX:MySQL 基于锁级别又分为:共享(读)锁、排他(写)锁、意向共享(读)锁、意向排他(写)锁
共享(读)锁、排他(写)锁、意向共享(读)锁、意向排他(写)锁
XX:对于共享(读)锁、排他(写)锁,比如咱们住酒店,入住前顾客都是有权看房的,只看不住想白嫖都是可以的,前台小姐姐会把门给你打开。当然,也允许不同的顾客一起看(共享 读),比如和这位杀马特小伙子。
看房时房间相当于公共场所,小姐姐嘱咐不能乱涂乱画,也不能偷喝免费的矿泉水。。如果你觉得不错,偷偷跑到前台要定这间房,交钱后会给你这个房间的钥匙并将房间状态改为已入住,不再允许其他人看房(排他 写)。
对了,当办理入住时前台小姐姐也会通知看房的杀马特小伙子说这间房已经有人定了!!等看房的杀马特小伙儿骂骂咧咧出门后,看到满头大汗的你,鄙夷着咽了一口口水,咳 tui!然后你锁上门哼着歌儿,开始干那些见不得人的事儿~~直到你退房前,其他人无法在看你的房。
可见,读锁是可以并发获取的(共享的),而写锁只能给一个事务处理(排他的)。当你想获取写锁时,需要等待之前的读锁都释放后方可加写锁;而当你想获取读锁时,只要数据没有被写锁锁住,你都可以获取到读锁,然后去看房。
另外还有意向读\写锁,严格来说他们并不是一种锁,而是存放表中所有行锁的信息。就像我们在酒店,当我们预定一个房间时,就对该行(房间)添加 意向写锁,但是同时会在酒店的前台对该行(房间)做一个信息登记(旅客姓名、男女、住多长时间、家里几头牛等)。大家可以把意向锁当成这个酒店前台,它并不是真正意义上的锁(钥匙),它维护表中每行的加锁信息,是共用的。后续的旅客通过酒店前台来看哪个房间是可选的,那么,如果没有意图锁,会出现什么情况呢?假设我要住房间,那么我每次都要到每一个房间看看这个房间有没有住人,显然这样做的效率是很低下的。杀马特小伙儿表示支持!
读写锁、意向锁的兼容性如下所示;
XX:再回到 MySQL 原理上讲
1 共享(读)锁(Share Lock)
共享锁,又叫读锁,是读取操作(SELECT)时创建的锁。其他用户可以并发读取数据,但在读锁未释放前,也就是查询事务结束前,任何事务都不能对数据进行修改(获取数据上的写锁),直到已释放所有读锁。
如果事务 A 对数据 B(1024 房)加上读锁后,则其他事务只能对数据 B 上加读锁,不能加写锁。获得读锁的事务只能读数据,不能修改数据。
SQL 显示加锁写法:
在查询语句后面增加 LOCK IN SHARE MODE,MySQL 就会对查询结果中的每行都加读锁,当没有其他线程对查询结果集中的任何一行使用写锁时,可以成功申请读锁,否则会被阻塞。其他线程也可以读取使用了读锁的表,而且这些线程读取的是同一个版本的数据。
2 排他(写)锁(Exclusive Lock)
排他锁又称写锁、独占锁,如果事务 A 对数据 B 加上写锁后,则其他事务不能再对数据 B 加任何类型的锁。获得写锁的事务既能读数据,又能修改数据。
SQL 显示加锁写法:
在查询语句后面增加 FOR UPDATE,MySQL 就会对查询结果中的每行都加写锁,当没有其他线程对查询结果集中的任何一行使用写锁时,可以成功申请写锁,否则会被阻塞。另外成功申请写锁后,也要先等待该事务前的读锁释放才能操作。
3 意向锁(Intention Lock)
意向锁属于表级锁,其设计目的主要是为了在一个事务中揭示下一行将要被请求锁的类型。InnoDB 中的两个表锁:
意向共享锁(IS):表示事务准备给数据行加入共享锁,也就是说一个数据行加共享锁前必须先取得该表的 IS 锁;
意向排他锁(IX):类似上面,表示事务准备给数据行加入排他锁,说明事务在一个数据行加排他锁前必须先取得该表的 IX 锁。
意向锁是 InnoDB 自动加的,不需要用户干预。
再强调一下,对于 INSERT、UPDATE 和 DELETE,InnoDB 会自动给涉及的数据加排他锁;对于一般的 SELECT 语句,InnoDB 不会加任何锁,事务可以通过以下语句显式加共享锁或排他锁。
共享锁:SELECT … LOCK IN SHARE MODE;
排他锁:SELECT … FOR UPDATE;
面试官:(这小子有两下子)嗯,元芳你怎么看?
HR:通俗易懂,我听懂了~~
面试官:好,那最后一个问题,你上面提到了乐观锁和悲观锁,谈谈你对它的看法吧。
XX:其实悲观锁和乐观锁,也并不是 MySQL 或者数据库中独有的概念,而是并发编程的基本概念。主要区别在于,操作共享数据时,“悲观锁”即认为数据出现冲突的可能性更大,而“乐观锁”则是认为大部分情况不会出现冲突,进而决定是否采取排他性措施。
反映到 MySQL 数据库应用开发中,悲观锁一般就是利用类似 SELECT … FOR UPDATE 这样的语句,对数据加锁,避免其他事务意外修改数据。乐观锁则与 Java 并发包中的 AtomicFieldUpdater 类似,也是利用 CAS 机制,并不会对数据加锁,而是通过对比数据的时间戳或者版本号,来实现乐观锁需要的版本判断。
MySQL 的多版本并发控制 (MVCC),其本质就可以看作是种乐观锁机制,而排他性的读写锁、两阶段锁等则是悲观锁的实现。
面试官:好,小 XX 我看你对 MySQL 锁这块儿确实研究得比较透彻,连 HR 都听懂了,还是让我比较满意的。回去等通知吧!!!
评论