java面试mysql慢查询_原创 记录一次线上Mysql慢查询问题排查过程

背景

前段时间收到运维反馈,线上mysql数据库凌晨时候出现慢查询的报警,并把原始sql发了过来:

--去除了业务含义的sql

update test_user set a=1 where id=1;

表数据量200w左右,不是很大,而且是根据主键更新。

问题排查

排查mysql数据库

我看到sql后第一反应就是是不是数据库出问题了,每个小时都有业务,偏偏白天业务高峰时间段正常,凌晨业务量很少时候出问题,让运维先检查了数据库的状态,反馈是数据库正常。

排查业务代码(第一次)

这块业务代码比较复杂,而且是别人写的,第一次看都没看完,直接在代码里打印了各个模块执行的时间,然后上线。

排查业务代码(第二次)

第二天又出现慢查询了,我赶紧下载了线上日志,发现整个方法执行时间很长,然后发现执行时间长的代码有几行调用其他服务的代码,使用的是httpclient,猜到了原因,应该是调用其他超时导致的。

说下系统整体流程,微信(a)回调我们的收银台服务(b),收银台更新订单信息并调用业务服务(c)。

出问题原因是:

第一次a调用b,b锁住记录行并调用c,这个时候c没有响应,导致a又发送了第二次请求。

第二次a调用b,b更新记录时候发生死锁,出现慢查询。

解决方案

收银台系统b接收回调的方法添加分布式锁,保证同一时刻同一订单只能更新一次。

收银台调用业务服务使用的是httpclient4.4,默认超时时间60秒,这么长时间如果对方没有响应就完了,改成了5秒,超时立马返回,不影响其他业务。

httpclient4.4版本设置超时时间代码如下:

httppost httppost = new httppost(url);

requestconfig.custom().setsockettimeout(5000).setconnecttimeout(5000).setconnectionrequesttimeout(5000).build();

httppost.setconfig(requestconfig);

上面设置了3个超时时间,含义如下:

setconnecttimeout:设置连接超时时间,单位毫秒。

setconnectionrequesttimeout:设置从connect manager获取connection 超时时间,单位毫秒。这个属性是新加的属性,因为目前版本是可以共享连接池的。

setsockettimeout:请求获取数据的超时时间,单位毫秒。 如果访问一个接口,多少时间内无法返回数据,就直接放弃此次调用。

最后,按照这两个方案改造上线后,系统运行正常,问题解决。

推荐阅读

如果觉得文章不错,希望可以随手转发或者”在看“哦,非常感谢哈!

关注下方公众号后回复「1024」,有惊喜哦!

希望与广大网友互动??

点此进行留言吧!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值