搜索
首页数据库mysql教程MySQL的表级锁,行级锁,排它锁和共享锁是什么

    前言

    由于我们的业务比较复杂,一个组成事务的相关多个SQL语句是必要的。因此,先解释什么是事务。一个事务是指一组SQL语句一起执行,必须要么全部执行成功,要么全部执行失败,不允许存在部分成功或部分失败的情况。一个事务有ACID特性:

    • 原子性:要么全部成功,要么全部失败,这样才能保证事务的一致性;

    • 一致性:比如银行的转账,扣除一个人的钱肯定要给另一个人加钱,不能光扣除不加,这样业务就存在问题,数据的一致性就破坏了;

    • 持久性:当我们数据commit以后,数据是先写到缓存当中,缓存中的数据还是要慢慢花时间往磁盘上写,如果此时停电了、宕机或者重启了,我们有redo log重做日志来保证数据库的持久性;

    • 隔离性:这块可以说下事务为什么要有隔离性,因为事务要允许并发执行,一个业务涉及了很多事务,而我们后台往往有很多业务,要能够让他们并发执行,如果所有的事务都是串行执行的话,那这样我们写多线程程序只有一个线程来做事情,这样效率很低。所以事务要并发执行,但是并发执行涉及了一些问题:事务的安全性&一致性并发的效率问题,我们以这两个东西为参考点,才得到了MySQL不同等级的并发/隔离,如果事务并发执行时我们完全不隔离的话,就可能会出现脏读(事务B读到了事务A还未提交的数据然后,然后用事务A未提交的数据去做计算,得到了很多其他的结果,然后事务A又把那个数据rollback掉,那么事务B计算出来的都是有问题的数据,脏读一定会出现问题)、不可重复读(以同样的条件去一个数据,然后再次去查询的时候发现数据的值有所改变,当然不可重复读也不一定会有问题,有些业务场景下是允许的,这和业务上数据的安全性和一致性是否严格有关)和幻读(在事务中按照同样的条件前后两次查询的结果数据量不同)这些问题。

    那么我们为了解决事务并发执行遇到的问题就给出了事务的隔离级别:

    • 串行化,串行化完全用锁来实现,通过锁给所有事务排序,按顺序执行,这样做数据的安全性高但并发的效率很低,一般我们不会这样做的。

    • 未提交读,对于我们写的多线程程序来说,对于临界区代码段没有做任何的并发控制,虽然并发性高但数据安全性很低,未提交读还允许脏读的存在,这是有问题的所以绝对不会使用未提交读。串行化和未提交读在实际项目中是不会用到的,一般数据库引擎默认工作在已提交读和可重复读,这两个隔离级别就结合了数据的安全性&一致性和数据的并发效率,这两个是由MVCC多版本并发控制机制实现的

    • 已提交读,oracle默认工作级别。不允许读取未commit的数据,这个级别仍然允许不可重复读和虚读产生。

    • 可重复读,MySQL默认工作级别。保证事务再次读取是依然得到相同的数据,部分解决了虚读,但虚读是仍然会出现的

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    注意:

    • 事务隔离级别越高,为避免冲突所花费的性能也就越多,即效率低。

    • 在“可重复读”级别,实际上可以解决部分的虚读问题,但是不能防止update更新产生的虚读问题,要禁止虚读产生,还是需要设置串行化隔离级别。

    事务隔离级别的实现原理:锁+MVCC。串行化底层实现原理是锁,锁有共享锁、排它锁、意向共享锁、意向排它锁、间隙锁和死锁,InnoDB的已提交读和可重复读的底层实现原理:MVCC(多版本并发控制),MVCC提供了一种并发读取方式,包括快照读(同一份数据会有多个版本)、当前读、undo log和redo log。MVCC是已提交读和可重复读的原理,锁是串行化的原理

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    事务日志被用于实现ACID特性,而共享锁、排它锁和MVCC则被用于实现一致性(I)特性。事务日志分为undo log(回滚日志) 和 redo log(重做日志)

    一、表级锁&行级锁

    • 表级锁:对整张表加锁。开销小(因为不用去找表的某一行的记录进行加锁,要修改这张表,直接申请加这张表的锁),加锁快,不会出

    • 现死锁;锁粒度大,发生锁冲突的概率高,并发度低

    • 行级锁:对某行记录加锁。开销大(需要找到表中相应的记录,有搜表搜索引的过程),加锁慢,会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度高

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    MyISAM存储引擎只支持表级锁,InnoDB支持事务处理,支持行级锁,并发能力更好

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    二、排它锁&共享锁

    • 排它锁:又称为X锁,写锁

    • 共享锁:又称为S锁,读锁

    读读(SS)之间是可以兼容的,但是读写(SX、SX)之间,写写(XX)之间是互斥的

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    1. 测试不同事务之间排它锁和共享锁的兼容性

    我们先查看表SQL及内容

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    查看隔离级别:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    首先开启一个事务A,给id=7的数据加上排它锁:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    在另一个客户端开启事务B:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    给id=7不管加排它锁和共享锁都阻塞了并没有查询出来,因为A事务给id=7这一行的数据加了排它锁,就是写锁,其他人不能读也不能写。

    总结:不同事务之间对于数据的锁,只有SS锁可以共存,XX、SX、XS都不能共存

    2. 测试行锁加在索引项上

    其实行锁是加在索引树上的。

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    每次做完测试都把刚做的rollback。

    用表的无索引字段作为过滤条件

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    那现在事务2获取不同行chenwei的记录

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    InnoDB是支持行锁的,刚才以主键id为过滤条件时,事务1和事务2获取不同行的锁是可以成功的。然而现在我们发现获取name为chenwei的排它锁也获取不到了,这是为什么?我们解释一下:

    InnoDB的行锁是通过给索引项加锁来实现的,而不是给表的行记录加锁实现的

    而我们用name作为过滤条件没有用到索引,自然就不会使用行锁,而是使用表锁。这就意味着只有通过索引检索数据,InnoDB才使用行级锁,否则InnoDB都将使用表锁!!!

    我们给name字段加上索引:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    然后再做刚才的操作:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    我们发现,给name加上索引后,两个事务可以获取到不同行的排它锁(for update),再一次证明了InnoDB的行锁是加在索引项上的。

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    因为现在name走的是索引, 通过zhangsan在辅助索引树上找到它所在行记录的id是7,然后到主键索引树上,获取对应行记录的排他锁(个人猜测应该是辅助索引树和主键索引树相应的记录都加了锁)

    三、串行化隔离级别测试

    串行化所有事务用的都是共享锁或者排它锁,不需用手动添加。select获取的是共享锁,insert、delete和update获取的都是排它锁。

    设置串行化隔离级别:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    两个事务可以同时获取共享锁(SS共存:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    现在让事务2插入数据;

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    由于Insert需要加排它锁,但是由于事务1已经对整张表加了共享锁,事务2无法再对表成功加锁(sx不共存)

    rollback一下,把所有获取锁的状态都回退掉:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    开启两个事务:

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    因为我们给name加上了索引,以上的select相当于给name为zhangsan的数据加上了行共享锁

    事务2update;

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    事务2不能update,因为此时已经被事务1的共享锁锁住了整个表

    事务2在辅助索引树上找zhangsan,找到对应的主键值,然后去主键索引树找到相应的记录,但是发现这行记录已经被共享锁锁住了,事务2可以获取共享锁,但是不能获取排他锁

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    我们再用主键索引试试id能不能update

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    依然阻塞住了,虽然我们where后面的字段现在使用的id而不是name,但是name也是通过辅助索引树找到对应的主键,再到主键索引树上找相应的记录,而主键索引树上的记录加了锁

    我们update id=8的数据,成功了。因为我们select的时候,只是给id=7的数据加上了行锁,我们操作id=8的数据当然可以成功

    MySQL的表级锁,行级锁,排它锁和共享锁是什么

    有索引,则使用行锁;没有索引,则使用表锁。

    表级锁还是行级锁说的是锁的粒度,共享锁和排他锁说的是锁的性质,不管是表锁还是行锁,都有共享锁和排他锁的区分。

    串行化玩的就是排它锁和共享锁,在可重复读级别下,不手动加锁的话,用的就是MVCC机制,实际上并没有用到锁,我们也可以手动加锁。InnoDB如果不创建索引的话,用的是表锁,如果查询的时候用到了索引项,它用的就是行锁了,行锁是给索引加锁,而不是单纯给一行数据加锁。

    以上是MySQL的表级锁,行级锁,排它锁和共享锁是什么的详细内容。更多信息请关注PHP中文网其他相关文章!

    声明
    本文转载于:亿速云。如有侵权,请联系admin@php.cn删除
    图文详解mysql架构原理图文详解mysql架构原理May 17, 2022 pm 05:54 PM

    本篇文章给大家带来了关于mysql的相关知识,其中主要介绍了关于架构原理的相关内容,MySQL Server架构自顶向下大致可以分网络连接层、服务层、存储引擎层和系统文件层,下面一起来看一下,希望对大家有帮助。

    mysql的msi与zip版本有什么区别mysql的msi与zip版本有什么区别May 16, 2022 pm 04:33 PM

    mysql的msi与zip版本的区别:1、zip包含的安装程序是一种主动安装,而msi包含的是被installer所用的安装文件以提交请求的方式安装;2、zip是一种数据压缩和文档存储的文件格式,msi是微软格式的安装包。

    mysql怎么去掉第一个字符mysql怎么去掉第一个字符May 19, 2022 am 10:21 AM

    方法:1、利用right函数,语法为“update 表名 set 指定字段 = right(指定字段, length(指定字段)-1)...”;2、利用substring函数,语法为“select substring(指定字段,2)..”。

    mysql怎么替换换行符mysql怎么替换换行符Apr 18, 2022 pm 03:14 PM

    在mysql中,可以利用char()和REPLACE()函数来替换换行符;REPLACE()函数可以用新字符串替换列中的换行符,而换行符可使用“char(13)”来表示,语法为“replace(字段名,char(13),'新字符串') ”。

    mysql怎么将varchar转换为int类型mysql怎么将varchar转换为int类型May 12, 2022 pm 04:51 PM

    转换方法:1、利用cast函数,语法“select * from 表名 order by cast(字段名 as SIGNED)”;2、利用“select * from 表名 order by CONVERT(字段名,SIGNED)”语句。

    MySQL复制技术之异步复制和半同步复制MySQL复制技术之异步复制和半同步复制Apr 25, 2022 pm 07:21 PM

    本篇文章给大家带来了关于mysql的相关知识,其中主要介绍了关于MySQL复制技术的相关问题,包括了异步复制、半同步复制等等内容,下面一起来看一下,希望对大家有帮助。

    mysql怎么判断是否是数字类型mysql怎么判断是否是数字类型May 16, 2022 am 10:09 AM

    在mysql中,可以利用REGEXP运算符判断数据是否是数字类型,语法为“String REGEXP '[^0-9.]'”;该运算符是正则表达式的缩写,若数据字符中含有数字时,返回的结果是true,反之返回的结果是false。

    带你把MySQL索引吃透了带你把MySQL索引吃透了Apr 22, 2022 am 11:48 AM

    本篇文章给大家带来了关于mysql的相关知识,其中主要介绍了mysql高级篇的一些问题,包括了索引是什么、索引底层实现等等问题,下面一起来看一下,希望对大家有帮助。

    See all articles

    热AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智能驱动的应用程序,用于创建逼真的裸体照片

    AI Clothes Remover

    AI Clothes Remover

    用于从照片中去除衣服的在线人工智能工具。

    Undress AI Tool

    Undress AI Tool

    免费脱衣服图片

    Clothoff.io

    Clothoff.io

    AI脱衣机

    AI Hentai Generator

    AI Hentai Generator

    免费生成ai无尽的。

    热门文章

    R.E.P.O.能量晶体解释及其做什么(黄色晶体)
    3 周前By尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O.最佳图形设置
    3 周前By尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O.如果您听不到任何人,如何修复音频
    3 周前By尊渡假赌尊渡假赌尊渡假赌

    热工具

    SublimeText3 Mac版

    SublimeText3 Mac版

    神级代码编辑软件(SublimeText3)

    PhpStorm Mac 版本

    PhpStorm Mac 版本

    最新(2018.2.1 )专业的PHP集成开发工具

    WebStorm Mac版

    WebStorm Mac版

    好用的JavaScript开发工具

    ZendStudio 13.5.1 Mac

    ZendStudio 13.5.1 Mac

    功能强大的PHP集成开发环境

    MinGW - 适用于 Windows 的极简 GNU

    MinGW - 适用于 Windows 的极简 GNU

    这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。