白话Mysql的锁和事务隔离级别!死锁、间隙锁你都知道吗?

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 我们把那些可能会被多个线程同时操作的资源称为临界资源,加锁的目的就是让这些临界资源在同一时刻只能有一个线程可以访问。这是当时在讲synchronized锁时提出的锁的概念。数据库作为用户共享的一个资源,如何保证数据并发访问一致性也是所有数据库必须解决的问题,如何加锁是数据库并发访问性能的一个重要因素。

听说微信搜索《Java鱼仔》会变更强哦!


本文收录于JavaStarter ,里面有我完整的Java系列文章,学习或面试都可以看看哦


(一)概述


我们把那些可能会被多个线程同时操作的资源称为临界资源,加锁的目的就是让这些临界资源在同一时刻只能有一个线程可以访问。这是当时在讲synchronized锁时提出的锁的概念。


数据库作为用户共享的一个资源,如何保证数据并发访问一致性也是所有数据库必须解决的问题,如何加锁是数据库并发访问性能的一个重要因素。


(二)关于数据库的锁


从加锁形式上分为乐观锁悲观锁


从对数据库操作的类型分为读锁(共享锁)写锁(排他锁)


读锁:多个读操作可以同时进行不会互相影响写锁:当前写操作没有完成前,会排斥其他的读锁和写锁


从数据操作粒度上可以分为表锁行锁


2.1 表锁


顾名思义,表锁就是锁住整张表。这种操作开销小,加锁快,不会出现死锁,但是锁的粒度大,并发度低。MyISAM引擎在操作数据时就会自动给数据加上表锁。


可以通过下面几个参数手动增加表锁:


lock table 表名 read(write)

查看表上加过的锁:


show open tables;

网络异常,图片无法展示
|


删除锁:


unlock tables;

加了读锁后,将限制写入;加了写锁之后,将限制读和写;


2.2 行锁


行锁就是锁住一行数据,这种做法开销大,加锁慢,会出现死锁。但是锁的粒度小,并发度最高。InnoDB支持行锁,同时InnoDB还支持事务


通过一个简单的例子来介绍一下行锁,我们开启两个session(navicat中就是开启两个查询窗口)去操作数据库,其中在第一个查询中修改数据,但是不提交:



begin;update test_innodb set name='javayz2'where id=1;

这个时候用另外一个查询窗口去修改同一行数据:


update test_innodb set name='javayz3'where id=1;

这时候这行更新语句会被一直阻塞,现在把第一个查询中修改的数据提交:


commit

你会发现第二个查询中的修改也立刻生效了: 查看结果,阻塞了30秒


网络异常,图片无法展示
|


注意:InnoDB的行锁是加在索引上的,如果索引失效或者修改的是非索引字段,行锁就会升级为表锁。


(三)Mysql的事务隔离级别


在讲事务隔离级别之前需要确保知道以下几个概念:


事务的ACID属性:原子性、一致性、隔离性、持久性


并发可能带来的问题:更新丢失、脏读、不可重复读、幻读


更新丢失:即多个事务同时更新一行数据,导致最后的更新覆盖了前面事务的更新。脏读:即事务A读取到了事务B已修改但是未提交的数据。 不可重复读:即事务A读取某些数据后再读取这些数据,发现这些数据发生了改变。因为事务A读取到了其他事务提交的数据,不符合隔离性。幻读:即事务A读取某些数据后按相同条件再读取发现多了新的数据,也是因为事务A读取到了其他事务新增的数据,不符合隔离性。


了解上面的概念之后,就可以知道Mysql的事务隔离级别了。事务隔离级别分为四种,下面通过表格的形式进行展示:



隔离级别
脏读

不可重复读

幻读
读未提交(read-uncommitted)
读已提交(read- committed) ×
可重复读(repeatable- read) × ×
可串行化(Serializable) × × ×

隔离级别越严格,并发带来的问题就越小,但是并发度也越低。


Mysql默认的事务隔离级别是可重复读。通过下列语句可查看


show VARIABLES like'tx_isolation'

通过下列语句修改事务隔离级别:


set tx_isolation='READ-COMMITTED'//READ-UNCOMMITTED、READ-COMMITTED、REPEATABLE-READ、SERIALIZABLE

(四)MVCC机制


当事务隔离级别设置为可重复读时,事务A执行select之后,如果事务B修改了数据,事务A再select依然是上一次的数据,但是如果用insert、update和delete时,采用的数据就会是事务B修改后的数据,


比如事务A第一次查询到的age是23、事务B将age加1,事务A第二次查询age依然是23;如果事务A将age加1,再查询就是25。


可重复读的隔离级别采用的是MVCC(multi-version concurrency control)的机制,它的实现机制比较复杂,我后续会出一篇专门的文章来讲一下。他所实现的目的是select操作读取的是历史的快照版本,insert、update、delete更新的是最新的版本


(五)可重复读的情况下如何解决幻读问题


Mysql默认的事务隔离级别是可重复读,但是没有解决幻读的问题,如果改为可串行化,并发度又太低,因此能否在可重复读的情况下解决幻读问题也是面试中比较常遇见的。

解决办法是用间隙锁。在其中一个事务中执行update语句时,增加一个范围,比如:



update test_innodb set name='javayz3'where id>1and id<10

id在1到10之间的数据就无法在其他事务中被插入或修改,自然也不会出现幻读的情况。


(六)关于死锁


在项目中偶尔会遇到死锁的情况,死锁的产生其实很简单,sessionA在等待sessionB的完成,sessionB在等待sessionA的完成,形成死循环。比如下面的更新语句:



A:begin;update test_innodb set name='aa'where id=1;//锁住id为1的行
B:begin;update test_innodb set name='bb'where id=2;//锁住id为2的行  
A:update test_innodb set name='aa'where id=2;B:update test_innodb set name='bb'where id=1;

最终A被B阻塞,B被A阻塞,从而导致了死锁。


网络异常,图片无法展示
|


Mysql中增加了死锁处理机制,对于死锁他会通过rollback解锁,但是有时遇到复杂的死锁情况就无法解锁。


(七)总结


整篇文章首先讲了Mysql的锁,Myisam使用表锁,InnoDB使用行锁。接着介绍了事务的基本原理,从而带出Mysql的事务隔离级别。简单介绍了MVCC机制,然后对于间隙锁、死锁进行讲解。掌握这些能让你在工作中或面试中遇到Mysql的问题有解决的思路。好了,这就是本期的所有内容了,我们下期再见!



相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
打赏
0
0
0
0
10
分享
相关文章
MySQL底层概述—9.ACID与事务
本文介绍了数据库事务的ACID特性(原子性、一致性、隔离性、持久性),以及事务控制的演进过程,包括排队、排它锁、读写锁和MVCC(多版本并发控制)。文章详细解释了每个特性的含义及其在MySQL中的实现方式,并探讨了事务隔离级别的类型及其实现机制。重点内容包括:ACID特性(原子性、持久性、隔离性和一致性的定义及其实现方式)、事务控制演进(从简单的全局排队到复杂的MVCC,逐步提升并发性能)、MVCC机制(通过undo log多版本链和Read View实现高效并发控制)、事务隔离级别(析了四种隔离级别(读未提交、读已提交、可重复读、可串行化)的特点及适用场景)、隔离级别与锁的关系。
MySQL底层概述—10.InnoDB锁机制
本文介绍了:锁概述、锁分类、全局锁实战、表级锁(偏读)实战、行级锁升级表级锁实战、间隙锁实战、临键锁实战、幻读演示和解决、行级锁(偏写)优化建议、乐观锁实战、行锁原理分析、死锁与解决方案
170 24
MySQL底层概述—10.InnoDB锁机制
京东面试:MySQL MVCC是如何实现的?如何通过MVCC实现读已提交、可重复读隔离级别的?
1.请解释什么是MVCC,它在数据库中的作用是什么? 2.在MySQL中,MVCC是如何实现的?请简述其工作原理。 3.MVCC是如何解决读-写和写-写冲突的? 4.在并发环境中,当多个事务同时读取同一行数据时,MVCC是如何保证每个事务看到的数据版本是一致的? 5.MVCC如何帮助提高数据库的并发性能?
京东面试:MySQL MVCC是如何实现的?如何通过MVCC实现读已提交、可重复读隔离级别的?
如何排查和解决PHP连接数据库MYSQL失败写锁的问题
通过本文的介绍,您可以系统地了解如何排查和解决PHP连接MySQL数据库失败及写锁问题。通过检查配置、确保服务启动、调整防火墙设置和用户权限,以及识别和解决长时间运行的事务和死锁问题,可以有效地保障应用的稳定运行。
182 25
MySQL事务日志-Undo Log工作原理分析
事务的持久性是交由Redo Log来保证,原子性则是交由Undo Log来保证。如果事务中的SQL执行到一半出现错误,需要把前面已经执行过的SQL撤销以达到原子性的目的,这个过程也叫做"回滚",所以Undo Log也叫回滚日志。
155 7
MySQL事务日志-Undo Log工作原理分析
MySQL进阶突击系列(06)MySQL有几种锁?| 别背答案,现场演示一下
本文详细解析了MySQL InnoDB存储引擎的锁机制,涵盖读锁、写锁、意向锁、记录锁、间隙锁和临键锁等8种锁类型。重点探讨了不同锁类型的加锁与释放方式,以及事务并发场景下的实战验证。通过具体示例,展示了在不同情况下锁的行为及其对事务的影响。文章还特别强调了锁的作用范围主要是索引,并解释了锁如何影响数据的读写操作。最后总结了并发事务中加锁规则,帮助读者深入理解MySQL的锁机制。
【MySQL基础篇】事务(事务操作、事务四大特性、并发事务问题、事务隔离级别)
事务是MySQL中一组不可分割的操作集合,确保所有操作要么全部成功,要么全部失败。本文利用SQL演示并总结了事务操作、事务四大特性、并发事务问题、事务隔离级别。
1769 2
【MySQL基础篇】事务(事务操作、事务四大特性、并发事务问题、事务隔离级别)
【YashanDB知识库】原生mysql驱动配置连接崖山数据库
【YashanDB知识库】原生mysql驱动配置连接崖山数据库
【YashanDB知识库】原生mysql驱动配置连接崖山数据库
docker拉取MySQL后数据库连接失败解决方案
通过以上方法,可以解决Docker中拉取MySQL镜像后数据库连接失败的常见问题。关键步骤包括确保容器正确启动、配置正确的环境变量、合理设置网络和权限,以及检查主机防火墙设置等。通过逐步排查,可以快速定位并解决连接问题,确保MySQL服务的正常使用。
277 82
大数据新视界--大数据大厂之MySQL数据库课程设计:MySQL 数据库 SQL 语句调优方法详解(2-1)
本文深入介绍 MySQL 数据库 SQL 语句调优方法。涵盖分析查询执行计划,如使用 EXPLAIN 命令及理解关键指标;优化查询语句结构,包括避免子查询、减少函数使用、合理用索引列及避免 “OR”。还介绍了索引类型知识,如 B 树索引、哈希索引等。结合与 MySQL 数据库课程设计相关文章,强调 SQL 语句调优重要性。为提升数据库性能提供实用方法,适合数据库管理员和开发人员。

推荐镜像

更多
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等