一条简单的 SQL 执行超过1000ms,纳尼?
liuian 2025-04-27 14:44 45 浏览
作者:VipAugus
https://juejin.im/post/5ce906a3e51d455a2f2201dc
MySQL对我说“Too young, too naive!"
大概过程
在测试环境Docker容器中,在跨进程调用服务的时候,A应用通过Dubbo调用B应用的RPC接口,发现B应用接口超时错误,接着通过debug和日志,发现具体耗时的地方在于一句简单SQL执行,但是耗时超过1000ms。
通过查看数据库的进程列表,发现是有死锁锁表了,很多进程状态status处于'sending data',最后为锁住的表添加索引,并且kill掉阻塞的请求,解除死锁,服务速度恢复正常。
下面记录的是大致排查过程:
通过观察业务代码,确认没有内存溢出或者其它事务问题,于是只能考虑Docker环境的数据库和jvm底层详情了。
使用Druid监控SQL执行状态
通过日志,发现有一句SQL严重超时,一句简单SQL,原本是批量插入多条记录,为了定位问题,测试时
Mybatis
只插入一条记录,但即便如此,还是耗时10秒
于是打算使用阿里巴巴的数据库连接池
Druid
进行监控,监控SQL效果如下:
在SQL监控Tab中,可以看到执行SQL的具体情况,包括某条SQL语句执行的时间(平均、最慢)、SQL执行次数、SQL执行出错的次数等。
上面显示的是正常情况下,时间单位是ms,正常的SQL一般在10ms之内,数据量大的控制在30ms之内,这样用户的使用体验感才会良好。所以说之前的1000ms,是不可接受的结果。
通过JMC远程监控Tomcat
JMC(java mission control)是jdk自带的一个监控工具,在jdk的bin目录下(java大法好,该目录下有很多实用的工具)。
此处加了一个tomcat无验证模式:
#在tomcat的conf目录下的catalina.sh增加如下java启动参数:-Dcom.sun.management.jmxremote=true-Dcom.sun.management.jmxremote.port=8888-Dcom.sun.management.jmxremote.ssl=false-Dcom.sun.management.jmxremote.authenticate=false-XX:+UnlockCommercialFeatures -XX:+FlightRecorder
下面是自己本地调试的截图
然后打开jmc,创建一个JMX连接,输入对应的ip和JMX端口。接着可以设定一段时间内的飞行监控,监测这一分钟内jvm具体参数
当时调试的时候,发现内存使用、CPU占用率、线程状态也挺正常的,没有发现明显的异常错误,效果如下图:
唯一比较耗时的是在代码tab页中,当时发现了大量的I/O,比上图的比例还高,当时大概占了80%,查看调用树,很多循环tcp socket连接,考虑到应用中本来就有很多需要io以及netty也需要tcp连接,所以大概排除了jvm虚拟机的问题,然后就去排查MySQL的问题。
排查MySQL
在了解MySQL锁概念的时候,由于现在使用的比较多的是InnoDB,所以可以着重看看InnoDB锁问题。推荐:
收藏起来,史上最全的 MySQL 高性能优化实战总结!
直接执行SQL语句
通过DEBUG代码,从mybatis中取出映射后的SQL语句,在MySQL客户款直接执行SQL和Explain查看执行计划,速度都很快,排除了SQL语句的问题。推荐:
值得收藏:一份非常完整的 MySQL 规范
查看MySQL线程列表
show processlist;
从图中可以看出,有些线程的状态处于sending data,查阅资料:所谓的“Sending data”并不是单纯的发送数据,而是包括“收集 + 发送 数据”。
然后后面一列info显示的是具体信息,是查询用来生成主键ID的函数,之前速度都很快,为啥突然就这么慢呢,于是回过头去查看该函数:
select next_value into ret_val from `xxx` where table_name=tableName for update;update `xxx` set current_value=current_value+step, next_value=next_value+stepwhere table_name=tableName;
select for update,给这个表加了排它锁,阻止其它事务取得相同数据集的共享读锁和排他写锁,同时,这个序列表表中,用来检索的字段没有加索引,在InnoDB行锁机制中:
由于MySQL的行锁是针对索引加的锁,不是针对记录加的锁,所以虽然是访问不同行的记录,但是如果是使用相同的索引键(在我们的场景中,就是查询时用到的table_name),是会出现锁冲突的。
所以了解到其它团队因为查询这个表产生事务问题,造成死锁,这个序列表被锁住了。
由于这个自增序列表每个团队都在使用,所以当时测试环境中,经常有dao层超时错误,最终将这些阻塞的线程kill掉,为序列表加了索引,解决了问题。
小结
下次遇到MySQL执行耗时的情况,排除了代码问题之后,要去看数据库是否有死锁的情况存在,观察有没有被阻塞的线程,排查被阻塞的线程具体info,定位到具体问题。
相关推荐
-
- win7激活错误0 80072f8f(win7激活时发生错误,错误代码0x80072f8f)
-
可以通过修改时间来解决系统提示0x80072F8F错误代码的操作方法。操作步骤1、鼠标点击时间,选择更改日期和时间设置,方法如下。2、在窗口中,选择internet时间,点击更改设置,方法如下。3、接着在窗口中,勾选与internet时间服...
-
2026-01-17 03:37 liuian
- 电脑如何进入bios模式(电脑怎么样进入bios)
-
电脑进入BIOS的方法1、按住F1电脑品牌:IBM、东芝、脸型thinkpad部分型号、toshiba等等这些型号电脑在开机时,按住键盘上的F1按键,那么它就会进入到BIOS的设置洁面中,需注意thi...
- 电脑如何设置虚拟内存(电脑怎么设虚拟内存怎么设最好)
-
进入系统打开控制面板后展开所有控制面板项,点击进入系统。修改虚拟内存点击高级系统设置后点击性能下的设置,点击高级后更改虚拟内存。设置数值将虚拟内存设置为自定义大小,具体数值最好是实际内存容量的一点五倍...
- 我的电脑图标打不开(电脑上图标消失了怎么恢复)
-
如果电脑点击图标没反应,首先看一下是不是鼠标左键的问题,左键出现故障图标往往是没有反应的,其次你把这些图标连一下,你看能不能画起来,如果无法划起来就是鼠标坏了,相反如果鼠标可以操作,但是电脑无法连接着...
- 联想y系列和r系列(联想7000)
-
联想y系列和r系列都好。其中r系列搭载的一般都是AMD锐龙处理器,y系列一般搭载的是英特尔处理器。希望我的回答对你有帮助一般情况下的话,联想笔记本电脑它的R系列和wy系列的画相比较来说是y系列,要稍微...
- 不激活win7有什么影响(不激活windows7会怎样)
-
如果Windows7未激活,会出现以下几种情况:1.桌面背景将变成黑色,每隔一段时间就会自动恢复成黑色。2.会弹出提示框,提醒用户需要激活Windows7。3.无法更新Windows,包括安...
- 超强升级系统 小说(超强升级系统 小说免费阅读)
-
女主柳洛溪,宋千千,肖甜甜,叶紫嫣,伏芸珊,南宫燕,菲菲,媛灵。狂暴系统的拥有者,龙家崛起的推动者,龙族守护者,天武大陆上屠魔者,新任远古界界王。前世为远古世界第一人,与远古世界一同孕育...
- win11暂停更新点不了(win11设置永不更新)
-
win11更新不能暂停了是设置错误导致的,解决方法如下1、首先右键单击此电脑,选择管理。2、然后在中点击服务和应用程序。3、在弹出的列表中继续点击服务。4、在服务的右侧找到windowsupdate...
- win10专业版下载速度慢(电脑win10下载速度慢)
-
由于是官方网址同时浏览该网址的网民非常多由于传输速率一定导致每个人的下载速率会变慢,可以错开高峰期如中午等,可以等到晚上或者早上期间进行下载。Win10系统浏览网页慢解决办法:1.在win10系统桌...
- 电脑关机总是关不掉(电脑关机一直关不掉怎么办)
-
电脑怎么关不了机是为什么电脑无法关机是因为还有软件在后台运行干预导致。鼠标右键点击计算机,选择属性,左侧菜单中选择高级系统设置,点击高级选项卡界面下方,启动和故障修复下的设置,取消勾选自动重新启动,点...
-
- apple id是什么(appleID是什么意思)
-
AppleID是苹果公司为其产品,如iWork、iTunesStore和AppleStore所引入的认证系统,使用Apple各项服务所需的用户名,就像是一把钥匙一样。有了AppleID,才能在苹果手机、平板电脑等设备上,使用苹...
-
2026-01-17 01:05 liuian
- 一键重装系统的软件有哪些(一键重装系统软件排行榜)
-
具体如下:一、韩博士装机大师韩博士装机大师是一款傻瓜式一键装系统工具,一键智能重装原版系统,无需光驱,无需U盘,零技术基础,傻瓜式操作,流水线流程,智能安装,完美支持GPT与win10平板!一款功能强...
- 电脑无法访问u盘(电脑无法访问u盘如何解决)
-
原因1:U盘被隐藏解决方法:1、在电脑上插入U盘后,依次打开“我的电脑”/“此电脑”-查看-选项。2、进入高级设置框后,点击“查看”,取消勾选“隐藏空的驱动器”项,最后点击“应用”即可。原因2:硬...
- u盘损坏如何恢复(u盘损坏怎么恢复里面的资料)
-
U盘损坏的五种常见问题修复方法,第一种是重新焊接,第二种加固,第三种,将数据转移,第四种利用软件修复,第五种采取U盘转换器进行转换。解决方法: (1)运行“强力数据恢复软件”,点击‘快速扫描’或‘深...
- 一周热门
-
-
飞牛OS入门安装遇到问题,如何解决?
-
如何在 iPhone 和 Android 上恢复已删除的抖音消息
-
Boost高性能并发无锁队列指南:boost::lockfree::queue
-
大模型手册: 保姆级用CherryStudio知识库
-
用什么工具在Win中查看8G大的log文件?
-
如何在 Windows 10 或 11 上通过命令行安装 Node.js 和 NPM
-
威联通NAS安装阿里云盘WebDAV服务并添加到Infuse
-
Trae IDE 如何与 GitHub 无缝对接?
-
idea插件之maven search(工欲善其事,必先利其器)
-
如何修改图片拍摄日期?快速修改图片拍摄日期的6种方法
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
