概述

数据库锁定机制简单的来说,就是数据库为了保证数据的一致性与完整性,而使各种共享资源在被并发访问时变得有序所设计的一种规则。对于任何一种数据库来说都需要有相应的锁机制,所以MySQL也不能例外。MySQL数据库根据锁锁定数据的颗粒度可分为表级锁、行级锁和页级锁。

一、表级锁

1.1 什么是表级锁

表级锁会直接锁定整张表。表级锁是MySQL各存储引擎中最大颗粒度的锁定机制。该锁定机制最大的特点是实现逻辑简单,带来的系统负面影响最小。所以获取锁和释放锁的速度很快。由于表级锁一次会将整个表锁定,所以可以很好的避免死锁问题。当然,锁定颗粒度大所带来负面影响就是出现争用锁定资源的概率也会最高,致使并发度大大降低。

1.2 表级锁的锁模式

MySQL的表级锁有两种模式:表共享读锁和表独占写锁;表共享读锁,不会阻塞其他用户对同一表的读请求,但会阻塞对同一表的写请求;表独占写锁,会阻塞其他用户对同一表的读和写操作;表共享读锁与表独占写锁之间,以及表独占写锁之间是串行的。当一个线程获得对一个表的写锁后,只有持有锁的线程可以对表进行更新操作。其他线程的读、写操作都会等待,直到锁被释放为止。

1.3 表锁优化建议

虽然使用表级锁定在锁定实现的过程中比实现行级锁定或者页级锁所带来的附加成本都要小,锁定本身所消耗的资源也最少。但是由于锁定的颗粒度比较大,所以造成争用锁定资源的情况也会比其他的锁级别都要多,从而在较大程度上会降低并发处理能力。所以,在优化表级锁问题的时候,最关键的就是如何让其提高并发度。由于锁定级别是不可能改变的了,所以我们首先需要尽可能让锁定的时间变短,然后就是让可能并发进行的操作尽可能的并发。

二、行级锁

2.1 什么是行级锁

行级锁仅对指定的记录加锁。行级锁最大的特点就是锁定对象的颗粒度小,也是目前各大数据库管理软件所实现的锁定颗粒度最小的。由于锁定颗粒度很小,所以发生争用锁定资源的概率也最小,能尽可能大的提高数据库的并发处理能力从而提高高并发应用系统的性能。虽然行级锁能够极大提高并发处理能力,但是由于锁定资源的颗粒度很小,所以每次获取锁和释放锁需要做的事情也更多,带来的消耗自然也就更大了。此外,行级锁定也最容易发生死锁。

2.2 行级锁的锁模式

MySQL行级锁有共享锁和排他锁两种。当一个事务需要给自己需要的某个资源加锁的时候,如果遇到一个共享锁正锁定着自己需要的资源的时候,自己可以再加一个共享锁,不过不能加排他锁。但是,如果遇到自己需要锁定的资源已经被一个排他锁占有之后,则只能等待该锁定释放资源之后自己才能获取锁定资源并添加自己的锁定。如果一个事务请求的锁模式与当前的锁兼容,InnoDB就将请求的锁授予该事务;反之,如果两者不兼容,该事务就要等待锁释放。

2.3 行锁优化建议

InnoDB行锁是通过给索引上的索引项加锁来实现的,只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁。在实际应用中,要特别注意InnoDB行锁的这一特性,不然的话,可能导致大量的锁冲突,从而影响并发性能。下面通过一些实际例子来加以说明。

1、在不通过索引条件查询的时候,InnoDB确实使用的是表锁,而不是行锁;

2、由于MySQL的行锁是针对索引加的锁,不是针对记录加的锁,所以虽然是访问不同行的记录,但是如果是使用相同的查询条件,是会出现锁冲突的;

3、当表有多个索引的时候,不同的事务可以使用不同的索引锁定不同的行,另外,不论是使用主键索引、唯一索引或普通索引,InnoDB都会使用行锁来对数据加锁;

4、即便在条件中使用了索引字段,但是否使用索引来检索数据是由MySQL通过判断不同执行计划的代价来决定的,如果MySQL认为全表扫描效率更高,比如对一些很小的表,它就不会使用索引,这种情况下InnoDB将使用表锁,而不是行锁;

2.4 间隙锁

当我们用范围条件而不是相等条件检索数据并请求共享锁或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但并不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个“间隙”加锁,这种锁机制就是间隙锁。

假如userInfo表中只有50条记录,其userId的值分别是1,2,3,....,49,50;

下面的SQL:

mysql> select * from emp where userId > 49 for update;

是一个范围条件的检索,InnoDB不仅会对符合条件的userId值为50的记录加锁,也会对userId大于50(这些记录并不存在)的“间隙”加锁。

InnoDB使用间隙锁的目的是防止幻读,以满足相关隔离级别的要求。对于上面的例子,如果不使用间隙锁,若其他事务插入了userId大于50的任何记录,那么本事务如果再次执行上述语句,就会发生幻读;因为间隙锁锁定的是一个范围,而不是具体的索引键所以在使用范围条件检索并锁定记录时,间隙锁会将不存在的键值也会被无辜的锁定,而造成在锁定的时候无法插入锁定键值范围内的任何数据。因此,在实际应用开发中,尤其是并发插入比较多的应用,我们要尽量优化业务逻辑,尽量使用相等条件来访问更新数据,避免使用范围条件。还要特别说明的是,InnoDB除了通过范围条件加锁时使用间隙锁外,如果使用相等条件请求给一个不存在的记录加锁,InnoDB也会使用间隙锁。

2.5 行锁优化建议

1、尽量让所有的查询条件都通过索引来完成,从而避免InnoDB因为无法通过索引键加锁而升级为表级锁定;

2、合理设计索引,让InnoDB在索引键上面加锁的时候尽可能准确,尽可能的缩小锁定范围,避免造成不必要的锁定而影响其它SQL的执行;

3、减少基于范围的数据检索过滤条件,避免因为间隙锁带来的负面影响而锁定了不该锁定的记录;

4、尽量控制事务的大小,减少锁定的资源量和锁定时间长度;

5、在业务环境允许的情况下,可使用较低级别的事务隔离,以减少MySQL因为实现事务隔离级别所带来的附加成本;

6、类似业务模块中,尽量按照相同的访问顺序来访问,防止产生死锁;

7、在同一个事务中,尽量做到一次锁定所需要的所有资源,减少死锁产生概率;

8、对于非常容易产生死锁的业务部分,可以尝试使用升级锁定颗粒度,通过表级锁定来减少死锁产生的概率;

三、页级锁

页级锁是MySQL中比较独特的一种锁定级别,在其他数据库管理软件中并不是太常见。页级锁定的特点是锁定颗粒度介于行级锁定与表级锁之间,所以获取锁定所需要的资源开销,以及所能提供的并发处理能力也同样是介于上面二者之间。另外,页级锁定和行级锁定一样,会发生死锁。

在数据库实现资源锁定的过程中,随着锁定资源颗粒度的减小,锁定相同数据量的数据所需要消耗的内存数量是越来越多的,实现算法也会越来越复杂。不过,随着锁定资源颗粒度的减小,应用程序的访问请求遇到锁等待的可能性也会随之降低,系统整体并发度也随之提升。

总的来说,MySQL这3种锁的特性可大致归纳如下:

表级锁:开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低;

行级锁:开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度也最高;

页面锁:开销和加锁时间界于表锁和行锁之间;会出现死锁;锁定粒度界于表锁和行锁之间,并发度一般。

四、行级锁和表级锁能共存吗?

行级锁与表级锁是可以共存的。InnoDB为了让表锁和行锁共存而引入了意向锁。

举个例子(此时假设行锁和表锁能共存): 事务A锁住表中的一行(写锁)。事务B锁住整个表(写锁)。

但你就会发现一个很明显的问题,事务A既然锁住了某一行,其他事务就不可能修改这一行。这与"事务B锁住整个表就能修改表中的任意一行"形成了冲突。所以,没有意向锁的时候,行锁与表锁共存就会存在问题!

那么意向锁是如何让表锁和行锁共存的?

有了意向锁之后,前面例子中的事务A在申请行锁(写锁)之前,数据库会自动先给事务A申请表的意向排他锁。当事务B去申请表的写锁时就会失败,因为表上有意向排他锁之后事务B申请表的写锁时会被阻塞。由此我们可得出意向锁是表锁。

(更多文章可关注微信公众号:IT鸡窝)

参考文章:https://blog.csdn.net/zcl_love_wx/article/details/82015281https://www.cnblogs.com/sessionbest/articles/8689071.html

乐观锁的颗粒度_MySql数据库锁机制详解相关推荐

  1. 乐观锁的颗粒度_mysql「锁」

    表锁:表级别的锁定是mysql各存储引擎中最大粒度的锁定机制,该锁定机制最大的特点是实现逻辑非常简单,带来的系统负面影响最小,所以获取锁释放锁的速度很快,由于表级锁一次会将整个表锁定,所以可以很好的避 ...

  2. MySQL锁、事务隔离级别、MVCC机制详解、间隙锁、死锁等

    一. 简介 1. 锁定义 锁是计算机协调多个进程或线程并发访问某一资源的机制. 在数据库中,除了传统的计算资源(如CPU.RAM.I/O等)的争用以外,数据也是一种供需要用户共享的资源.如何保证数据并 ...

  3. mysql数据库字段类型大全_mysql数据库字段类型详解

    MySQL支持大量的列类型,它可以被分为3类:数字类型.日期和时间类型以及字符串(字符)类型.本节首先给出可用类型的一个概述,并且总结每个列类型的存储需求,然后提供每个类中的类型性质的更详细的描述. ...

  4. mysql source 数据库_MySQL 数据库 source 命令详解及实例

    MySQL 数据库 source 命令详解及实例 MySQL 数据库 source 命令,该命令是数据库导入命令.source 命令的用法非常简单,首先你需要进入 MySQL 数据库的命令行管理界面, ...

  5. mysql数据库导出_MySQL数据库导入导出详解[转发]

    1. 概述 MySQL数据库的导入,有两种方法: 1) 先导出数据库SQL脚本,再导入: 2) 直接拷贝数据库目录和文件. 在不同操作系统或MySQL版本情况下,直接拷贝文件的方法可能会有不兼容的情况 ...

  6. mysql数据库导入导出_MySQL数据库导入导出详解

    MySQL数据库的导入,有两种方法:1) 先导出数据库SQL脚本,再导入:2) 直接拷贝数据库目录和文件.在不同操作系统或MySQL版本情况 1. 概述 MySQL数据库的导入,有两种方法: 1) 先 ...

  7. mysql数据库数字类型_Mysql 数据库数值类型详解

    MySQL 支持所有标准SQL 中的数值类型,其中包括严格数值类型(INTEGER.SMALLINT.DECIMAL 和NUMERIC),以及近似数值数据类型(FLOAT.REAL 和DOUBLE P ...

  8. mysql数据库教程全解_MySQL数据库基础教程详解

    为了确保数据的完整性和一致性,在创建表时指定字段名称,字段类型和字段属性外,还需要使用约束(constraint),索引(index),主键(primary key)和外键(foregin key)等 ...

  9. Mysql查询汉字语法_Mysql数据库查询语法详解

    ___聚合函数___max():最大值min():最小值avg():平均值sum():和count():记数 group_concat():组内字段拼接,用来查看组内其他字段 ___example__ ...

最新文章

  1. DOM对象和内置对象(中)
  2. 188tokens_Affiliate 追踪技术: tokens 和 postback URL 设置的教程
  3. java 代码块的作用_Java核心(三):代码块的作用
  4. 【mongodb系统学习之六】mongodb配置文件方式启动
  5. c/c++函数指针(Hook前奏1)
  6. linux内核剖析(八)进程间通信之-管道
  7. SAP Fiori issue排查:why search by ID does not work
  8. scanf( )函数的返回值
  9. 小说中场景的功能_场景在小说中的作用
  10. python random
  11. ubuntu上装texlive
  12. python在线朗读-简单的python代码实现语音朗读
  13. 解析大型.NET ERP系统 单据标准(新增,修改,删除,复制,打印)功能程序设计...
  14. 敏捷项目管理之计划扑克游戏
  15. 蓝桥杯特殊回文数C语言简易版
  16. 关于Coolie编码
  17. 远行星号java 出错_远行星号跳出问题[已解决]
  18. JAVA EE面试重点
  19. C语言求一元二次方程的根,这题很简单嘛?看看这种想法很惊奇!
  20. android工程师考司考,法考真题app哪个好用_司考刷题哪个app好用_搜法考真题app哪个比较好用...

热门文章

  1. xstream不映射字段_这本XStream学习手册,真的不来看看?
  2. win10桌面倒计时插件_win10 桌面如何做到清爽好看?这篇教程给你答案
  3. 阿里妈妈技术团队 6 篇论文入选 CIKM 2021
  4. 【阿里妈妈数据科学系列】第三篇:离线抽样框架下的AB Test
  5. 统计问题:找链环数字对的出现频率
  6. abaqus python二次开发攻略_基于CAE二次开发进行弹簧批量建模
  7. 深度学习(6)TensorFlow基础操作二: 创建Tensor
  8. linux 上传网页,Windows和Linux主机FTP上传网页注意事项
  9. 『设计模式』小伙你的穿搭很潮!--装饰者模式
  10. 『设计模式』就因为多收了我2块5,我追着收银员问是不是不懂设计模式--策略模式