当前位置:网站首页>深入理解Mysql锁与事务隔离级别
深入理解Mysql锁与事务隔离级别
2022-07-07 21:51:00 【打分几楼MOTO】
目录
查看INFORMATION_ SCHEMA系统库锁相关数据表
Mysql事务及ACID特性详解
概述
我们的数据库一般都会并发执行多个事务, 多个事务可能会并发的对相同的一批数据进行增删改查操作, 可能就会导致我们说的脏写、脏读.不可重复读、幻读这些问题。
这些问题的本质都是数据库的多事务并发问题,为了解决多事务并发问题,数据库设计了事务隔离机制、锁机制、MVCC多版本并发控制隔离机制, 用一整套机制来解决多事务并发问题。接下来,我们会深入讲解这些机制,让大家彻底理解数据库内部的执行原理。
事务及其ACID属性
事务是由一组SQL语句组成的逻辑处理单元,事务具有以下4个属性通常简称为事务的ACID属性。
原子性(Atomicity):事务是一个原子操作单元,其对数据的修改要么全都执行,要么全都不执行。
一致性(Consistent): 在事务开始和完成时,数据都必须保持一致状态。 这意味着所有相关的数据规则都必须应用于事务的修改,以保持数据的完整性。
隔离性(solation) :数据库系统提供一定的隔离机制,保证事务在不受外部并发操作影响的“ 独立环境执行。这意味着事务处理过程中的中间状态对外部是不可见的.反之亦然。
持久性(Durable) :事务完成之后,它对于数据的修改是永久性的.即使出现系统故障也能够保持。
并发事务处理带来的问题
更新丢失(Lost Update)或脏写
当两个或多个事务选择同一行, 然后基于最初选定的值更新该行时,由于每个事务都不知道其他事务的存在,就会发生丢失更新问题一最后的更新覆盖了由其他事务所做的更新。
脏读(Dirty Reads)
一个事务正在对一条记录做修改,在这个事务完成并提交前,这条记录的数据就处于不一致的状态;这时,另一个事务也来读取同一条记录,如果不加控制,第二个事务读取了这些“脏“数据,并据此作进一步的处理, 就会产生未提交的数据依赖关系。这种现象被形象的叫做“脏读"。
一句话:事务A读取到了事务B已经修改但尚未提交的数据,还在这个数据基础上做了操作。此时,如果B事务回滚, A读取的数据无效,不符合一致性要求。
不可重复读(Non-Repeatable Reads)
一个事务在读取某些数据后的某个时间,再次读取以前读过的数据,却发现其读出的数据已经发生了改变.或某些记录已经被删除了!这种现象就叫做不可重复读”。
一句话:事务A内部的相同查询语句在不同时刻读出的结果不一致,不符合隔离性
幻读(Phantom Reads)
一个事务按相同的查询条件重新读取以前检索过的数据,却发现其他事务插入了满足其查询条件的新数据,这种现象就称为“幻读"。
Mysql事务隔离级别详解
Mysql事务隔离级别详解
脏读,不可重复读和幻读,其实都是数据库读一致性问题,必须有数据库提供一定的事务隔离机制来解决
数据库的事务隔离越严格,并发副作用越小,但付出的代价也就越大,因为事务隔离实质上就是使事务在一定程度上“串行化“进行,这显然与"并发”是矛盾的。
同时,不同的应用对读一致性和事务隔离程度的要 求也是不同的,比如许多应用对“不可重复读"和“幻读并不敏感,可能更关心数据并发访问的能力。
查看当前数据库的事务隔离级别: show variables like 'tx_ _isolation';
设置事务隔离级别: set tx_ jisolation="REPEATABLE-READ';
Mysq|默认的事务隔离级别是可重复读,用Spring开发程序时,如果不设置隔离级别默认用Mysq|设置的隔离级别,如果Spring设置了就用已经设置的隔离级别
Mysql锁机制详解
锁详解
锁是计算机协调多个进程或线程并发访问某一资源的机制。在数据库中,除了传统的计算资源(如CPU. RAM. I/0等) 的争用以外,数据也是一种供需 要用户共享的资源。如何保证数据并发访问的一致性、有效性是所有数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素。
锁分类
从性能上分为乐观锁(用版本对比来实现)和悲观锁
从对数据库操作的类型分, 分为读锁和写锁(都属于悲观锁)
读锁(共享锁, S锁(Shared)) : 针对同一份数据,多个读操作可以同时进行而不会互相影响
写锁(排它锁,X锁(eXclusive) : 当前写操作没有完成前,它会阻断其他写锁和读锁
从对数据操作的粒度分, 分为表锁和行锁
表锁
每次操作锁住整张表,开销小加销快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低,一 般用在整表数据迁移的场景。
#手动增加表锁 lock table 表名称 read(write),,表名称2 read(write); #查看表上加过的锁 show open tables; #删除表锁 unlock tables;
案例分析(加读锁)
Lock table myLock read;
当前session和其他session都可以读该表
当前session中插入或者更新锁定的表都会报错,其他session插入或更新则会等待
案例分析(加写锁)
Lock table myLock write;
当前session对该表的增删改查都没有问题,其他session对该表的所有操作被阻塞
案例结论
对MyISAM表的读操作(加读锁) ,不会阻寒其他进程对同一表的读请求,但会阻赛对同一表的写请求。只有当读锁释放后,才会执行其它进程的写操作。
对MyISAM表的写操作(加写锁),会阻塞其他进程对同一表的读和写操作,只有当写锁释放后,才会执行其它进程的读写操作
行锁
每次操作锁住一行数据.开销大,加锁慢(先要找某一张表,再从表里找到某一行);会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度最高。
InnoDB与MYISAM的最大不同有两点:
InnoDB支持事务(TRANSACTION)
InnoDB支持行级锁
行锁演示
一个个session开启事务更新不提交,另一个session更新同一条记录会阻塞,更新不同记录不会阻塞
总结
MyISAM在执行查询语句SELECT前,会自动给涉及的所有表加读锁,在执行update、insert delete操作会自动给涉及的表加写锁。
InnoDB在执行查询语句SELECT时,因为有mvcc机制不会加锁。但是update、 insert. delete操作会加行锁。
简而言之,就是读锁会阻塞写,但是不会阻塞读。而写锁则会把读和写都阻塞。
间隙锁(Gap Lock)
- 间隙锁,锁的就是两个值之间的空隙。Mysql默认级别是repeatable-read, 有办法解决幻读问题吗?间隙锁在某些情况下可以解决幻读问题。假设account表里数据如下:
- 缺了3-10,10-20,20-正无穷三个区间,那么间隙就有id为(3,10), (10,20), (20,正无穷)
- 在Session_ 1下面执行update account set name = 'zhuge' where id > 8 and id <18;,则其他Session没法在这个范围所包含的所有行记录(不管这个行记录是否存在)以及范围内间隙行记录(不管这个行记录是否存在)所在的原表所有间隙里插入或修改任何数据,即id在(3,20]区间都无法修改数据, 注意最后那个20也是包含在内的。此时如果别的事务插入id在(3,20]的数据行,就会一致阻塞,解决了幻读
- 间隙锁是在可重复读隔离级别下才会生效。
临键锁(Next-key Locks)
- Next-Key Locks是行锁与间隙锁的组合。像上面那个例子里的这个(3,20]的整个区间可以叫做临键锁。
无索引行锁会升级为表锁
- 锁主要是加在索引上,如果对非索引字段更新,行锁可能会变表锁
- session1执行: update account set balance = 800 where name = lilei;如果name不是索引或者没有走索引,那么行锁就会变成表锁
- session2对该表任一行操作都会阻塞住
- InnoDB的行锁是针对索引加的锁,不是针对记录加的锁、并且该索引不能失效,否则都会从行锁升级为表锁.
- 锁定某一行还可以用lock in share mode(共享锁)和for update(排它锁),例如: select * from test_ innodb_ lock where a = 2 for update;这样其他session只能读这行数据,修改则会被阻塞,直到锁定行的session提交
结论
- Innodb存储引擎由于实现了行级锁定,虽然在锁定机制的实现方面所带来的性能损耗可能比表级锁定会要更高一下,但是在整体并发处理能力方面要远远优于MYISAM的表级锁定的。当系统并发量高的时候,Innodb的整体性能和MYISAM相比就会有比较明显的优势了。但是,Innodb的行级锁定同样也有其脆弱的-面,当我们使用不当的时候,可能会让Innodb的整体性能表现不仅不能比MYISAM高,甚至可能会更差。
行锁分析
- 通过检查InnoDB_ _row_ lock状态变量来分析系统上的行锁的争夺情况
show status like 'innodb_row_ 1ock%' ;
对各个状态量的说明如下:
- Innodb_ row_ lock_ current_ waits: 当前正在等待锁定的数量
- Innodb_ row_ lock_ _time: 从系统启动到现在锁定总时间长度
- Innodb_ row_ _lock_ time_ avg: 每次等待所花平均时间
- Innodb_ row_ lock_ time_ max: 从系统启动到现在等待最长的一次所花时间
- Innodb_ row_ _lock_ waits:系统启动后到现在总共等待的次数
对于这5个状态变量,比较重要的主要是:
- Innodb_ row_ lock_ _time_ _avg (等待 平均时长)
- Innodb_ row_ _lock_ _waits (等待总次数)
- Innodb_ row_ _lock_ time (等待总时长)
尤其是当等待次数很高,而且每次等待时长也不小的时候,我们就需要分析系统中为什么会有如此多的等待,然后根据分析结果着手制定优化计划。
查看INFORMATION_ SCHEMA系统库锁相关数据表
--查看事务
select * from INFORMATION_ SCHEMA. INNODB_ _TRX;
--查看锁
select * from INFORMATION_ SCHEMA. INNODB_ LOCKS;
--查看锁等待
select * from INFORMATION_ SCHEMA. INNODB_ _LOCK_ WAITS ;
--释放锁,trx_ mysq1_ _thread_ id可以从INNODB_ _TRX表里查看到
kill trx_ mysql_ _thread_ _id
死锁
- set tx_ isolation='repeatable-read;
- Session_1执行: select * from account where id=1 for update;
- Session_2执行: select * from account where id=2 for update;
- Session_1执行: select * from account where id=2 for update;
- Session_2执行: select * from account where id=1 for update;
- 查看近期死锁日志信息: show engine innodb status\G;
- 大多数情况mysq可以自动检测死锁并回滚产生死锁的那个事务,但是有些情况mysq没法自动检测死锁
锁优化建议
- 尽可能让所有数据检索都通过索引来完成,避免无索引行锁升级为表锁
- 合理设计索引,尽量缩小锁的范围
- 尽可能减少检索条件范围,避免间隙锁
- 间隙id为(3,10), (10,20), (20,正无穷) 这三个区间,在Session_ 1下面执行update account set name = 'zhuge' where id > 8 and id <25,如果检索条件范围是这个,那么锁的范围是(3,正无穷);如果在插入一个id为26的数据,间隙id为(3,10), (10,20), (20,26),(26,正无穷)那么锁的范围是(3,26],大大减少了锁的范围
- 尽量控制事务大小,减少锁定资源量和时间长度,涉及事务加锁的sq|尽量放在事务最后执行
- 尽可能低级别事务隔离
行锁与事务隔离级别案例分析
读未提交
打开一个客户端A,并设置当前事务模式为read uncomlmitted (未提交读),查询表account的初始值:
set tx_ isolation='read-uncommitted'
在客户端A的事务提交之前,打开另一个客户端B,更新表account:
这时,虽然客户端B的事务还没提交,但是客户端A就可以查询到B已经更新的数据
一旦客户端B的事务因为某种原因回滚,所有的操作都将会被撤销,那客户端A查询到的数据其实就是脏数据:
在客户端A执行更新语句update account set balance = balance - 50 where id =1. lilei的balance没有变成350, 居然是400,是不是很奇怪,数据不一致啊,如果你这么想就太天真了,在应用程序中,我们会用400-50=350, 并不知道其他会话回滚了,要想解决这个问题可以采用读已提交的隔离级别
由上述可以看出 事务A读取到了事务B已经修改但尚未提交的数据出现脏读;事务执行的两个时间点执行查询操作,数据不一致出现不可重复读;如果事务B插入一个新的数据事务A也可以读取到出现幻读
读已提交
- 打开一个客户端A,并设置当前事务模式为read committed (未提交读),查询表account的所有记录:set tx_ _isolation='read-committed';
- 在客户端A的事务提交之前,打开另一个客户端B,更新表account:
- 这时,客户端B的事务还没提交,客户端A不能查询到B已经更新的数据,解决了脏读问题:
- 客户端B的事务提交
- 客户端A执行与上一步相同的查询,结果与上一步不-致, 即产生了不可重复读的问题
可重复读
打开一个客户端A,并设置当前事务模式为repeatable read, 查询表account的所有记录
set tx_ isolationt'repeatable-read";
- 在客户端A的事务提交之前,打开另一个客户端B,更新表account并提交
- 在客户端A查询表account的所有记录, 与步骤(1) 查询结果-致,没有出现不可重复读的问题,但是此时数据库中的真实值是350,不管其他事务是不是做了修改,事务开始后是多少.在该事务结束之前如果没有对该值做修改仍然是多少
- 在客户端A,接着执行update account set balance = balance - 50 where id = 1, balance没有变成400-50=350,而是350-50=300.可见ilei的balance值用的是步骤2中的350来算的,所以是300,数据的一致性倒是没有被破坏。 可重复读的隔离级别下使用了MVCC(multi-version concurrency control)机制,select操作不会更新版本号,是快照读(历史版本) ; insert,update和delete会更新版本号, 是当前读(当前版本)。
- 重新打开客户端B,插入一条数据后提交 (注意此时客户端B还没有提交,所以id为1的数据行仍然没有改变,虽然事务A已经重新改变了这个值)
- 在客户端A查询表account的所有记录,没有查出新增数据,所以没有出现幻读
- 验证幻读:在客户端A执行update account set balance-888 where id = 4;能更新成功,再次查询能查到客户端B新增的数据(可重复读没有办法彻底解决幻读,select语句不会加锁)
可串行化
- 打开-个客户端A,并设置当前事务模式为serializable, 查询表account的初始值:
set tx_ isolation='serializable';
- 打开-个客户端B,并设置当前事务模式为serializable,更新相同的id为1的记录会被阻塞等待,更新id为2的记录可以成功, 说明在串行模式下innodb的查询也会被加上行锁; 如果客户端A执行的是一个范围查询, 那么该范围内的所有行包括每行记录所在的间隙区间范围(就算该行数据还未被插入也会加锁,这种是间隙锁)都会被加锁。此时如果客户端B在该范围内插入数据都会被阻塞,所以就避免了幻读。这种隔离级别并发性极低,开发中很少会用到。
- 注意,客串行是针对查询来的数据进行加锁,如果事务A进查询id为1的数据,那么事务B可以修改,增加别的数据行(id不为1),此时相当于事务B对这些数据行加锁;事务A是无法对这些数据进行查询,修改或者删除
边栏推荐
- JMeter-接口自动化测试读取用例,执行并结果回写
- Bit operation
- Solve the problem of duplicate request resource paths /o2o/shopadmin/o2o/shopadmin/getproductbyid
- Dynamic agent explanation (July 16, 2020)
- Adrnoid开发系列(二十五):使用AlertDialog创建各种类型的对话框
- Matlab-SEIR传染病模型预测
- 力扣解法汇总648-单词替换
- 七月第一周
- 网络安全-永恒之蓝
- 2021ICPC上海 H.Life is a Game Kruskal重构树
猜你喜欢
Install a new version of idea. Double click it to open it
Inftnews | web5 vs Web3: the future is a process, not a destination
LDO稳压芯片-内部框图及选型参数
Are the microorganisms in the intestines the same as those on the skin?
ArcGIS: field assignment_ The attribute table field calculator assigns values to fields based on conditions
微信论坛交流小程序系统毕业设计毕设(3)后台功能
微信论坛交流小程序系统毕业设计毕设(4)开题报告
云原生正在吞噬一切,开发者该如何应对?
Transform XL translation
Talk about the design and implementation logic of payment process
随机推荐
Introduction to anomaly detection
iNFTnews | NFT技术的广泛应用及其存在的问题
Wechat forum exchange applet system graduation design (5) assignment
网络安全-sqlmap与DVWA爆破
经纬度PLT文件格式说明
What are the similarities and differences between smart communities and smart cities
Add data analysis tools in Excel
Wechat forum exchange applet system graduation design completion (6) opening defense ppt
Are the microorganisms in the intestines the same as those on the skin?
GEE(四):计算两个变量(影像)之间的相关性并绘制散点图
iNFTnews | Web5 vs Web3:未来是一个过程,而不是目的地
LeeCode -- 6. Z 字形变换
Bea-3xxxxx error code
网络安全-安装CentOS
./ setup. Insufficient sh permission
Lecture 30 linear algebra Lecture 5 eigenvalues and eigenvectors
Clean C disk
海内外技术人们“看”音视频技术的未来
ArcGIS:矢量要素相同字段属性融合的两种方法
JMeter-接口自动化测试读取用例,执行并结果回写