浅谈MySQL 有哪些死锁场景

 更新时间:2022年08月11日 14:09:05   作者:萧木易  
本文主要介绍了浅谈MySQL 有哪些死锁场景,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

首先一起来复习一下死锁的概念:死锁是指两个或者多个事务在同一资源上相互占用,并请求锁定对方占用的资源,从而导致恶性循环的现象。

下面我们通过几个实验,来验证几种死锁场景。

1 环境准备

use martin;
drop table if exists dl;
CREATE TABLE `dl` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`a` int(11) NOT NULL,
`b` int(11) NOT NULL,
`c` int(11) NOT NULL,  
PRIMARY KEY (`id`),
KEY `idx_c` (`a`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
CREATE TABLE `dl_insert` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`a` int(11) NOT NULL,
`b` int(11) NOT NULL,
`c` int(11) NOT NULL,  
PRIMARY KEY (`id`),
unique key `uniq_a` (`a`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
insert into dl(a,b,c) values (1,1,1),(2,2,2);
drop table if exists dl_1;
create table dl_1  like dl;
insert into dl_1 select * from dl;

2 同一张表下的死锁

session1

session2

begin;

begin;

select * from dl where a=1 for update;…1 row in set (0.00 sec)

select * from dl where a=2 for update;…1 row in set (0.00 sec)

select * from dl where a=2 for update;/* SQL1 */(等待)

 

(session2 提示死锁回滚后,SQL1 成功返回结构)

select * from dl where a=1 for update;ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

commit;

commit;

session1 在等待 session2 释放 a=2 的行锁,而 session2 在等待 session1 释放 a=1 的行锁。两个 session 互相等待对方释放资源,就进入了死锁状态。

3 不同表下的死锁

session1

session2

begin;

begin;

select * from dl where a=1 for update; … 1 row in set (0.00 sec)

select * from dl_1 where a=1 for update; … 1 row in set (0.00 sec)

select * from dl_1 where a=1 for update;/* SQL2 */ 等待

 

(session2 提示死锁回滚后,SQL1 成功返回结构)

select * from dl where a=1 for update; ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

commit;

commit;

这个实验也是两个 session 互相等待对方释放资源,就进入了死锁状态。

4 间隙锁下的死锁

session1

session2

set session transaction_isolation='REPEATABLE-READ'; /* 设置会话隔离级别为 RR */

set session transaction_isolation='REPEATABLE-READ'; /* 设置会话隔离级别为 RR */

begin;

begin;

select * from dl where a=1 for update; … 1 row in set (0.00 sec)

select * from dl where a=2 for update; … 1 row in set (0.00 sec)

insert into dl(a,b,c) values (2,3,3);/* SQL1 */ 等待

 

(session2 提示死锁回滚后,SQL1 成功返回结果)

insert into dl(a,b,c) values (1,4,4);/* SQL2 */ ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

commit;

commit;

由于 RR 隔离级别下存在间隙锁,可以知道 SQL1 需要等待 a=2 获得的间隙锁,而 SQL2 需要等待 a=1 获得的间隙锁,两个 session 互相等待对方释放资源,就进入了死锁状态。

5 INSERT 语句的死锁

session1

session2

session3

begin;

  

insert into dl_insert(a,b,c) value (3,3,3);

  
 

insert into dl_insert(a,b,c) value (3,3,3);/* 等待 */

insert into dl_insert(a,b,c) value (3,3,3);/* 等待 */

rollback;

执行成功

ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction

这里需要注意的是,a 字段有唯一索引。当 session1 执行完 insert 语句,会在索引 a=3 上加记录锁,当 session2 执行同样的 insert 语句时,唯一键冲突,加上读锁;同样 session3 也会加上读锁。

当 session1 回滚,session2 和 session3 都试图继续执行插入操作,都要加上写锁。此时两个 session 都要等待对方的行锁,因此出现了死锁。

一些死锁场景就介绍到这里,当然,也欢迎各位补充其他的一些死锁场景。

到此这篇关于浅谈MySQL 有哪些死锁场景的文章就介绍到这了,更多相关MySQL 死锁场景内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MySQL多表查询机制

    MySQL多表查询机制

    这篇文章主要介绍了MySQL多表查询机制,多表查询首先离不开等值连接,下文我们从等值连接展开详细内容,具有一定的参考价值需要的小伙伴可以参考一下
    2022-03-03
  • Mysql数据库清理binlog日志命令详解

    Mysql数据库清理binlog日志命令详解

    这篇文章主要给大家介绍了Mysql数据库清理binlog日志命令的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Mysql具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-09-09
  • MySQL中配置文件my.cnf因权限问题导致无法启动的解决方法

    MySQL中配置文件my.cnf因权限问题导致无法启动的解决方法

    这篇文章主要给大家介绍了关于MySQL中配置文件my.cnf因权限问题导致无法启动的解决方法,该无法启动的错误提示代码是:World-writable config file '/etc/my.cnf' is ignored,文中给出了详细的解决方法,需要的朋友们下面来一起看看吧。
    2017-06-06
  • SQL SERVER 日期格式转换详解

    SQL SERVER 日期格式转换详解

    本篇文章是对SQL SERVER 日期格式转换进行了详细的分析介绍,需要的朋友参考下
    2013-06-06
  • 基于Redo Log和Undo Log的MySQL崩溃恢复解析

    基于Redo Log和Undo Log的MySQL崩溃恢复解析

    这篇文章主要介绍了基于Redo Log和Undo Log的MySQL崩溃恢复流程,点进来的小伙伴不要错过奥
    2021-08-08
  • 轻松掌握MySQL函数中的last_insert_id()

    轻松掌握MySQL函数中的last_insert_id()

    相信大家应该都知道Mysql函数可以实现许多我们需要的功能,这篇文章介绍的Mysql函数Last_insert_id()就是其中之一,文章通过一个例子展开来讲,应该更有助于大家的理解和学习,有需要的朋友们下面来一起看看吧。
    2016-12-12
  • 详解MySQL如何避免克隆失败后再次初始化

    详解MySQL如何避免克隆失败后再次初始化

    本文章讨论了当您没有足够的磁盘空间来存储两个数据集时,使用带有安全选项DATA DIRECTORY 的 CLONE INSTANCE 命令,所以接下来小编给大家详细的介绍一下,MySQL如何避免克隆失败后再次初始化,需要的朋友可以参考下
    2023-10-10
  • MySQL中DATEDIFF()函数用法及返回周、月、年

    MySQL中DATEDIFF()函数用法及返回周、月、年

    这篇文章主要给大家介绍了关于MySQL中DATEDIFF()函数用法及返回周、月、年的相关资料,datediff()函数返回两个日期之间的时间,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2023-11-11
  • 将MySQL从MyISAM转换成InnoDB错误和解决办法

    将MySQL从MyISAM转换成InnoDB错误和解决办法

    原来自己用的是为了装的, 所以在设置database usage(如下图1)的时候按照discuz官方的建议,选的都是Non-Transactional Database Only(只支持MyISAM数据引擎的非事务数据库),用MyISAM数据库,还没涉及到需要InnoDB,因此打算直接不加载INNODB引擎。
    2011-09-09
  • 如何利用Mysql计算地址经纬度距离实时位置

    如何利用Mysql计算地址经纬度距离实时位置

    最近工作中遇到了一个附近门店的功能,下面这篇文章主要给大家介绍了关于如何利用Mysql计算地址经纬度距离实时位置的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-04-04

最新评论