MySQL死锁问题案例

MySQL死锁问题

问题描述:在一张流水生成的记录表中,当没有当前条件的数据时候,并发情况下会导致有线程因为死锁问题生成流水号失败。

场景

有一张生成流水的表:

在这里插入图片描述
场景复现

简单来说,在根据流水类型、年、月、日生成流水信息的时候有则update流水号+1,无则insert;

Mapper

@Mapper
public interface NumberTestMapper {@Update("update number_test set serial_number = serial_number+1 where number_type = #{numberType} and number_year = #{numberYear} and number_month=#{numberMonth} and number_day = #{numberDay}")int update(NumberTest numberTest);@Insert("insert into number_test values (#{numberType},#{numberYear},#{numberMonth},#{numberDay},#{serialNumber})")int insert(NumberTest numberTest);@Select("select * from number_test where number_type = #{numberType} and number_year = #{numberYear} and number_month=#{numberMonth} and number_day = #{numberDay}")NumberTest selectById(NumberTest numberTest);
}

service

@Service
public class TestService {@Autowiredprivate NumberTestMapper mapper;@Transactionalpublic long generate() {NumberTest test = new NumberTest();test.setNumberType(1);test.setNumberYear("2024");test.setNumberMonth("01");test.setNumberDay("01");int update = mapper.update(test);if(update == 0){test.setSerialNumber(0);mapper.insert(test);}NumberTest select = mapper.selectById(test);return select.getSerialNumber();}
}

测试

@SpringBootTest
@RunWith(SpringRunner.class)
public class NumberTestMapperTest {@Autowiredprivate TestService service;@Testpublic void test() throws InterruptedException {List<Long> result = new CopyOnWriteArrayList<>();CountDownLatch countDownLatch = new CountDownLatch(5);ExecutorService executorService = Executors.newFixedThreadPool(5);for (int i = 0; i < 5; i++) {executorService.submit(() -> {try{result.add(service.generate());}catch (Exception e){e.printStackTrace();}finally {countDownLatch.countDown();}});}countDownLatch.await();executorService.shutdown();result.forEach(item -> System.out.println(item));}@Testpublic void test2() throws InterruptedException {CountDownLatch countDownLatch = new CountDownLatch(10);ExecutorService executorService = Executors.newFixedThreadPool(10);for (int i = 0; i < 10; i++) {executorService.submit(() -> {try{service.insert();}catch (Exception e){e.printStackTrace();}finally {countDownLatch.countDown();}});}countDownLatch.await();executorService.shutdown();System.out.println("success!");}
}

结果

测试在不存在当前行数据的情况下进行

  1. test方法会有死锁产生
  2. test2则报主键冲突问题

在这里插入图片描述
test方法为什么会产生死锁问题呢?

原因

开启两个事务分别执行:
T1:

BEGIN --step1
--step3
UPDATE number_test set serial_number = serial_number+1 where number_type = 1 and number_year = '2024' and number_month ='01' and number_day = '01';
--step5
insert into number_test VALUES (1,'2024','01','01',0);COMMIT

T2:

BEGIN --step2
--step4
UPDATE number_test set serial_number = serial_number+1 where number_type = 1 and number_year = '2024' and number_month ='01' and number_day = '01';
--step6
insert into number_test VALUES (1,'2024','01','01',0);
COMMIT

按照step1~step5执行发现:

  1. 执行step5的时候被阻塞,无法获取锁
  2. 执行step6的时候死锁产生
  3. 证明执行update的时候是有锁的,那锁的是什么呢?
  4. 经过多次实验发现,将insert中的number_type修改为2后不会去获取锁,证明update锁的是number_type字段。

将表结构修改:
在这里插入图片描述
然后发现:

  1. update没有使用id主键条件的时候是表锁,使用的时候是行锁
  2. 在加了联合索引后,问题仍然存在,而且是表锁,执行到step5后任何数据都插不进去
    在这里插入图片描述

总结:

在并发情况下,多个线程下执行update会有锁,但是update和update直接不会有锁,update和insert之间有锁。insert和update之间相互抢锁形成死锁。

这里不知道为什么在没有数据的时候,同时update相同条件的时候不会有锁,而执行insert的时候就有锁,推测是Mysql为了解决幻读问题使用间隙锁的原因。如果有大神知道,多谢指教

解决方案:

由于Mysql有解决死锁的机制了,采用try-catch重试解决这个问题

    @Transactionalpublic long generate() {try {NumberTest test = new NumberTest();test.setNumberType(1);test.setNumberYear("2024");test.setNumberMonth("01");test.setNumberDay("01");//update number_test set serial_number = serial_number+1 where number_type = #{numberType} and number_year = #{numberYear} and number_month=#{numberMonth} and number_day = #{numberDay}int update = mapper.update(test);if(update == 0){test.setSerialNumber(0);//insert into number_test values (#{numberType},#{numberYear},#{numberMonth},#{numberDay},#{serialNumber})mapper.insert(test);}NumberTest select = mapper.selectById(test);return select.getSerialNumber();}catch (Exception e){return generate();}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/51710.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python如何快速定位最慢的代码?优雅了~

编写Python代码时&#xff0c;我们常常会遇到性能瓶颈&#xff0c;这不仅影响程序的执行效率&#xff0c;还可能导致用户体验下降。那么&#xff0c;如何快速定位代码中最慢的部分&#xff0c;成为每个开发者必须掌握的技能。 如何快速定位 Python 代码中的性能瓶颈&#xff1…

MySql8.0在centos安装

在 CentOS 上安装 MySQL 8.0 的步骤如下&#xff1a; 添加 MySQL YUM 存储库&#xff1a; 首先&#xff0c;下载并添加 MySQL YUM 存储库到你的系统。 wget https://dev.mysql.com/get/mysql80-community-release-el7-3.noarch.rpm sudo rpm -Uvh mysql80-community-release-el…

Url图标实现

Url图标实现 效果如下&#xff1a; 1.引入样式 <link rel"icon" href"favicon.ico"> favicon.ico和对应的html一般需要在同一个目录下&#xff08;同级别&#xff09;。 2.title是用来设置在url页签中显示的名称。 可能存在的问题&#xff1a; …

Q15 三数之和

思路 我的思路&#xff0c;先排序&#xff0c;循环确定前两个值&#xff0c;最后一个值二分 看了题解&#xff1a; 最后两个值可以使用双指针&#xff0c;因为是有序的&#xff0c;而且是求和 注意重复值&#xff1a; public List<List<Integer>> threeSum(int[…

前端实现文本超出指定行数显示”展开”和”收起”效果

目录 效果演示步骤一&#xff1a;实现整体框架步骤二&#xff1a;实现样式步骤三&#xff1a;js实现元素控制完整代码 效果演示 本文方法是利用js原生进行实现的&#xff0c;可根据相关vue或react语法进行相关的改写&#xff0c;并实现效果 步骤一&#xff1a;实现整体框架 <…

OpenCV库学习之cv2.Sobel函数

OpenCV库学习之cv2.Sobel函数 一、简介 cv2.Sobel是OpenCV库中用于边缘检测的函数。它基于Sobel算子&#xff0c;通过计算图像在水平和垂直方向上的一阶导数来检测边缘。Sobel算子是一种离散差分算子&#xff0c;能够有效地突出图像中的高频变化区域&#xff0c;即边缘。 二、…

c-periphery RS485串口库文档serial.md(serial.h)(非阻塞读)(VMIN、VTIME)

c-peripheryhttps://github.com/vsergeev/c-periphery 文章目录 NAMESYNOPSISENUMERATIONS关于奇偶校验枚举类型 DESCRIPTIONserial_new()serial_open()关于流控制软件流控制&#xff08;XON/XOFF&#xff09;硬件流控制&#xff08;RTS/CTS&#xff09;选择流控制方法 serial_…

独立3D网络游戏《战域重甲》开发与上架经验分享

“ 小编阿麟&#xff1a;心之所向便是光&#xff0c;我们都是追光者!这位独立游戏开发者的产品能力已经不输给许多小团队&#xff0c;希望他的故事和经验分享&#xff0c;可以给走在同样道路上的朋友一些信心和帮助。 背景介绍 2023年年底的时候&#xff0c;我突然有一个很强的…

防火墙简单学习

文章目录 防火墙主要功能防火墙类型 防火墙主要功能 防火墙是一种网络安全设备&#xff0c;它通过监控和控制进出网络的数据包来保护内部网络不受外部攻击和威胁。防火墙的主要功能包括&#xff1a; 访问控制&#xff1a;防火墙可以限制哪些设备和用户可以访问网络资源&#x…

硬件工程师笔面试真题汇总

目录 1、电阻 1&#xff09;上拉电阻的作用 2&#xff09;PTC热敏电阻作为电源电路保险丝的工作原理 2、电容 1&#xff09;电容的特性 2) 电容的特性曲线 3) 1uf的电容通常来滤除什么频率的信号 3、电感 4、二极管 1&#xff09;二极管特性 2&#xff09;二极管伏安…

HVV | .NET 攻防工具库,值得您拥有!

01阅读须知 此文所提供的信息只为网络安全人员对自己所负责的网站、服务器等&#xff08;包括但不限于&#xff09;进行检测或维护参考&#xff0c;未经授权请勿利用文章中的技术资料对任何计算机系统进行入侵操作。利用此文所提供的信息而造成的直接或间接后果和损失&#xf…

软考高级第四版备考--第32天(新一代信息技术及应用)

1、物联网 1.1技术基础 1.1.1感知层&#xff1a;由各种传感器构成&#xff0c;包括温度传感器&#xff0c;二维码标签、RFID标签和读写器&#xff0c;摄像头&#xff0c;GPS等感知终端。感知层是物联网识别物体、采集信息的来源。 1.1.2网络层&#xff1a;由各种网络&#x…

《破解验证码:用Requests和Selenium实现模拟登录的终极指南》

两种模拟登录方式(图形验证码) 超级鹰 打码平台&#xff0c;用于识别验证码 requests模拟登录 from chaojiying import Chaojiying_Client import requests from requests import Session from lxml import etree #获取图片信息 def get_pic_info(img_name):chaojiying Ch…

10个append()函数在Python程序开发中的创新应用

文末赠免费精品编程资料~~ 在Python编程的世界里&#xff0c;append()函数是列表操作中最常见的方法之一。它允许我们在列表的末尾添加一个元素&#xff0c;这一简单的功能却能激发无限的创造力。今天&#xff0c;我们将探讨append()函数在Python程序开发中的10种创新应用&…

分布式文件存储行业解决方案和技术选型分析

分布式文件存储行业解决方案和技术选型分析 前言 上一集&#xff0c;我们已经完成了初始化测试报告以及判断压测类型的实战&#xff0c;我们在文章的末尾提到了文件上传的问题以及文件存储的问题&#xff0c;也说了接下来的几集中&#xff0c;我们会讨论分布式文件存储的内容…

代码随想录第23天|回溯

39.组合总和 题目链接/文章讲解&#xff1a; 代码随想录 视频讲解&#xff1a;带你学透回溯算法-组合总和&#xff08;对应「leetcode」力扣题目&#xff1a;39.组合总和&#xff09;| 回溯法精讲&#xff01;_哔哩哔哩_bilibili 第一想法&#xff1a; 组合总和与第22天组合总…

爬虫实战-掌上高考网实战

1.确定需求&#xff1a;爬取什么数据爬取大学名称 2.找到数据源地址数据在哪个链接中https://api.zjzw.cn/web/api/?keyword&page1&province_id&ranktype&request_type1&size20&top_school_id[3703,2461,659,3117,597,1724]&type&uriapidata/…

2024电赛H题参考方案——自动行使小车

目录 一、题目要求 二、参考资源获取 三、参考方案 1、环境搭建及工程移植 2、移植MPU6050模块 3、移植TB6612电机驱动模块 其他模块根据需要移植 总结 一、题目要求 小编自认为&#xff1a;此次H题属于控制类题目&#xff0c;相较于往年较为简单&#xff0c;功能也算单一&…

Vue3响应式高阶用法之toRaw()

Vue3响应式高阶用法之toRaw() 文章目录 Vue3响应式高阶用法之toRaw()一、简介二、使用场景2.1 性能优化2.2 与外部库的集成 三、基本使用3.1 创建响应式对象3.2 获取原始对象3.3 修改原始对象 四、功能详解4.1 toRaw的工作原理4.2 使用注意事项 五、最佳实践及案例5.1 性能优化…

基于深度学习的智能手势识别系统

基于深度学习的石头剪刀布手势识别&#xff08;UI界面YOLOv8/v7/v6/v5代码训练数据集&#xff09; 引言 石头剪刀布是一种简单而有趣的游戏。通过基于深度学习的手势识别系统&#xff0c;我们可以自动检测和识别玩家的手势。本文将详细介绍如何构建一个石头剪刀布手势识别系统…