一、锁介绍

不同存储引擎支持的锁是不同的,比如MyISAM只有表锁,而InnoDB既支持表锁又支持行锁。

下图展示了InnoDB不同锁类型之间的关系:

图中的概念比较多不好理解,下面依次进行说明。

1.1乐观锁

?乐观锁是相对悲观锁而言的,乐观锁假设数据一般情况下不会造成冲突,所在在数据进行提交更新时,才会对数据的冲突与否进行检测,如果发现冲突了,则返回给用户错误信息,让用户决定如何处理,其核心是基于CAS算法。乐观锁适用于读多写少的场景,可以提高程序吞吐量。

?Mysql自带的是没有乐观锁的,但是可以通过表上加个version字段来实现自己乐观锁。

假如要更新一个用户的年龄,可以这样做:

查出用户id等于3的用户信息,select id,name,age,version from user where id = 3,得到如下的数据。

id

Name

Age

Version

3

张三

26

1

更新张三的年龄为27,注意where条件带上版本号。update user set age = 27,version = 2 where id = 3 and version = 1;

如果更新的结果是1则表示更新成功了,如果是0则表示更新失败需要重新尝试。

1.2悲观锁

?悲伤锁就是在每次操作数据时,都悲观地认为会出现数据冲突,所以必须先获取到数据的锁再对其修改。传统的关系型数据库用的就是悲观锁,还有JDK中的synchronized关键字等。悲观锁主要分为共享锁和排他锁。

1.3共享锁

共享锁【shared locks】,又叫读锁,顾名思义,共享锁就是多个事务对同一个数据可以共享一把锁,都能访问到数据,但是只能读不能修改。

如何获取共享锁?

select * from user where id = 3 lock in share mode;

注意:在有事务获取到了共享锁之后,其他事务是不能做insert/update/delete操作的,因为insert/update/delete语句会自动加上排他锁。

1.4排他锁

排他锁【exclusive locks】,又叫写锁,顾名思义,排他锁就是不能与其他锁并存,如果一个事务获取了一个数据行的排他锁,其他事务就不能再获取该行的其他锁,包括共享锁和排他锁,但是获取排他锁的事务是可以对数据进行读取和修改。

如何获取排他锁?

在sql语句后加上for update即可。

select * from user where id = 3 for update

1.5表锁

表锁,顾名思义就是对整张表加锁,是Mysql各存储引擎中最大粒度的锁定机制。

优点:实现逻辑简单,获取锁和释放锁的速度很快,由于每次都是将整张表锁定所以可以很好的避免死锁问题。

缺点:锁定颗粒度大导致出现锁定资源争用的概率高,并发度低。

1.6行锁

行锁,顾名思义就是对表中的某行数据加锁,锁定颗粒度最小。

优点:发生锁冲突的概率低,并发处理能力强。

缺点:由于锁定资源的颗粒度很小,所以每次获取锁和释放锁需要做的事情也更多,带来的消耗自然也就更大了。此外,行级锁定也最容易发生死锁。

如何判断使用的是行锁还是表锁?

InnoDB的行锁是针对索引加的锁,不是针对记录加的锁,所以只有在通过索引条件检索数据时才会用行锁,否则使用表锁。并且该索引不能失效,否则都会从行锁升级为表锁。所以在使用select for update时,where 子句一定要带上索引,否则极容易造成性能问题。

行锁又细分三种实现算法:

record lock:专门对索引项加锁;

gap lock:间隙锁,是对索引之间的间隙加锁;

Next-key lock:是前面两种的组合,对索引及其之间的间隙加锁;

1.7页面锁

页面锁出现比较少,它的特点是开销和加锁时间界于表锁和行锁之间,会出现死锁,锁定粒度界于表锁和行锁之间,并发度一般。

二、死锁

2.1死锁原理

?死锁(Deadlock) 所谓死锁:是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程称为死锁进程。由于资源占用是互斥的,当某个进程提出申请资源后,使得有关进程在无外力协助下,永远分配不到必需的资源而无法继续运行,这就产生了一种特殊现象死锁。

死锁的四个必要条件:

互斥条件:一个资源每次只能被一个进程使用。

占有且等待:一个进程因请求资源而阻塞时,对已获得的资源保持不放。

不可强行占有:进程已获得的资源,在末使用完之前,不能强行剥夺。

循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。

2.2死锁案例

案例一

首先创建一张订单记录表,用于做订单的幂等性校验防止重复生成订单。

CREATE TABLE `order_record` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`order_no` int(11) DEFAULT NULL,

`status` int(4) DEFAULT NULL,

`create_date` datetime(0) DEFAULT NULL,

PRIMARY KEY (`id`) USING BTREE,

INDEX `idx_order_status`(`order_no`,`status`) USING BTREE

) ENGINE = InnoDB

事务A

事务B

关闭自动提交事务,set autocommit = 0;

set autocommit = 0;

select id from order_record where order_no = 4 for update;//检查是否存在订单号为4的订单

select id from order_record where order_no = 5 for update;//检查是否存在订单号为5的订单

//如果没有则插入信息

insert into order_record(order_no,status,create_date) values(4,1,‘2020-10-04 10:56:00‘);

此时锁等待中...

//如果没有则插入信息

insert into order_record(order_no,status,create_date) values(5,1,‘2020-10-04 10:56:00‘);

返回结果表明发生死锁,ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

COMMIT;(未完成)

COMMIT;(未完成)

分析:

由于order_no列为非唯一索引,而且此时是RR事务隔离级别,所以SELECT 的加锁类型是gap lock,而且gap范围是(4,+∞)。

当我们执行插入 SQL 时,会在插入间隙上再次获取插入意向锁。插入意向锁其实也是一种 gap 锁,它与 gap lock 是冲突的,事务 A 和事务 B 都持有间隙 (4,+∞)的 gap 锁,而接下来的插入操作为了获取到插入意向锁,都在等待对方事务的 gap 锁释放,于是就造成了循环等待,导致死锁。

案例二

InnoDB 存储引擎的主键索引为聚簇索引,其它索引为辅助索引。如果两个更新事务使用了不同的辅助索引,或者一个使用辅助索引,一个使用了聚簇索引,就都有可能导致锁资源的循环等待,造成死锁。

步骤:

首先,order_record表存在以下数据。

然后打开两个窗口

事务A

事务B

BEGIN;

BEGIN;

update order_record set status = 1 where order_no = 4;

mysql> update order_record set status = 1 where id = 4;

ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction//发生了死锁

分析:

事务A

事务B

首先获取idx_order_status辅助索引

获取主键索引的行锁

根据辅助索引获取主键索引,再获取主键索引的行锁

更新status列时,需要idx_order_status辅助索引

所以再更新数据时,要尽量根据主键来更新,可以有效避免死锁发生。

二、如何避免死锁

通常有以下手段可以预防死锁的发生:

在编程中尽量按照固定的顺序来处理数据库记录,假设有两个更新操作,分别更新两条相同的记录,但更新顺序不一样,有可能导致死锁。

在允许幻读和不可重复读的情况下,尽量使用 RC 事务隔离级别,可以避免 gap lock 导致的死锁问题。

更新表时,尽量使用主键更新。

避免长事务,尽量将长事务拆解,可以降低与其它事务发生冲突的概率;

设置锁等待超时参数,我们可以通过 innodb_lock_wait_timeout 设置合理的等待超时阈值,特别是在一些高并发的业务中,我们可以尽量将该值设置得小一些,避免大量事务等待,占用系统资源,造成严重的性能开销。

如果真的发生了数据库死锁,也有以下方式处理:

查看当前的事务

SELECT * FROM INFORMATION_SCHEMA.INNODB_TRX;

查看当前锁定的事务

SELECT * FROM INFORMATION_SCHEMA.INNODB_LOCKS;

查看当前等锁的事务

SELECT * FROM INFORMATION_SCHEMA.INNODB_LOCK_WAITS;

杀死进程 kill pid

而且MySQL默认开启了死锁检测机制,当检测到死锁后会选择一个最小(锁定资源最少的)的事务进行回滚。

三、总结

平常很少写MySQL相关的文章,其实MySQL中的门道还是挺多的,本文关于间隙锁等概念讲的比较简单,推荐博客《mysql间隙锁》。

以后可能会再写一篇关于索引的,也有可能不会(主要是懒??),如果本文哪里有错误,请多指教。

大话MySQL锁

mysql锁申请步骤_大话MySQL锁相关推荐

  1. mysql MDL锁如何解决_理解MySQL的MDL元数据锁

    一.MDL锁的作用 MySQL DBA 对于 Waiting for table metadata lock 肯定不会陌生,一般都是进行 alter 操作时被堵住了,导致了我们在 show proce ...

  2. mysql 无线网络设置_操作mysql命令行的方法步骤

    操作mysql命令行的方法步骤 现在向大家介绍mysql命令行下,从数据库的建立到表数据的删除全过程,希望对大家有所帮助. 登陆mysql 打cmd命令终端,如果已经添加了mysql的环境变量,可以直 ...

  3. mysql切换使用数据库_切换MySQL数据库步骤

    1.ABP官网下载对应的框架代码,并还原NuGet包 2.在控制台中程序包源选择 全部 默认程序选择 *.EntityFrameworkCore,并依次输入 1 install-package pom ...

  4. 一个项目部署多个节点会导致锁失效么_不为人知的分布式锁实现,全都在这里了

    1引入业务场景 首先来由一个场景引入: 最近老板接了一个大单子,允许在某终端设备安装我们的APP,终端设备厂商日活起码得几十万到百万级别,这个APP也是近期产品根据市场竞品分析设计出来的,几个小码农通 ...

  5. mysql数据库连接配置路径_[zz]MySQL数据库主从同步安装与配置总结

    注意:本文出自"阿飞"的博客 ,如果要转载本文章,请与作者联系! 并注明来源: http://blog.sina.com.cn/s/blog_49fd52cf0100pog2.ht ...

  6. mysql 中有什么命令_常用mysql命令大全

    常用的MySQL命令大全 连接MySQL格式: mysql -h主机地址 -u用户名 -p用户密码 1.例1:连接到本机上的MYSQL. 首先在打开DOS窗口,然后进入目录 mysqlbin,再键入命 ...

  7. mysql双机数据热备份_配置MySQL数据库双机热备份

    1.mysql 数据库没有增量备份的机制,当数据量太大的时候备份是一个很大的问题.还好 mysql 数据库提供了一种主从备份的机制,其实就是把主数据库的所有的数据同时写到备份数据库中.实现 mysql ...

  8. mysql 查询存储过程 速度_查询mysql过程

    MySql 使用explain分析查询 今天写了个慢到哭的查询,想用explain分析下执行计划,后来发现explain也是有局限性的: EXPLAIN不会告诉你关于触发器.存储过程的信息或用户自定义 ...

  9. mysql 集群操作系统_高性能MySQL集群详解(二)

    一.通过Keepalived搭建MySQL双主模式的高可用集群系统 1.MySQL Replication介绍: MySQL Replication是MySQL自身提供的一个主从复制功能,其实也就是一 ...

最新文章

  1. l293d电机驱动原理_一文搞懂步进电机特性、原理及驱动器设计
  2. I2S 和 PCM 区别
  3. IE6/7 BUG 图片不能居中
  4. dcdc芯片效率不高的原因_研学丨燃料电池车的典型效率及能耗
  5. python 串口助手 简书_python用pyserial读取串口问题解决
  6. mutilprocess模块的用法
  7. pcb板子制作各层的解释和用法
  8. oracle 64位数据源,64位Windows 7下32位 Oracle(含XE)ODBC 数据源的配置
  9. 烽火fr2600怎么web登录_烽火配置教程
  10. 如何将mp3转换flac音频文件格式
  11. 子曰:中庸之为德也,其至矣乎!民鲜久矣。
  12. Tomcat启动之后遇到“ran out of the normal time range, it consumed [2000] milliseconds.”?
  13. 最美的公式:你也能懂的麦克斯韦方程组
  14. CE是什么缩写,是什么含义
  15. HBase2.x完全分布式集群安装
  16. 第6章 冷眼看“学而无用”——《逆袭大学》连载
  17. 对中间层的一些浅略的思考
  18. 帝云CMS-免费可商用的万能PHP建站程序
  19. FTP工具,3款FTP工具推荐
  20. UNIX痛恨者手册[转贴自 FreeBSDChina]

热门文章

  1. 支付宝也跟上了!免费办理ETC 还提供设备包邮服务
  2. 雷军:技术立业是小米血液里最重要的东西
  3. 腾讯获准在中国销售Switch游戏机 任天堂股价应声飙升逾14%
  4. 周鸿祎:融360与360公司没有任何关系
  5. 虚拟资源拳王公社:最适合上班没时间的副业赚钱项目是什么,简单易操作的副业项目
  6. 拳王虚拟项目公社:2020主流的虚拟资源项目,最新最全自动化系统玩法
  7. JavaScript基础(五分钟让你了解js全貌)
  8. php foreach结果如何保存_每天一个PHP语法四引用使用及实现
  9. USB HID学习:一点开发记录
  10. 95-32-010-ChannelPipeline-ChannelPipeline简介