首页 > 其他分享 >delete语句执行过慢

delete语句执行过慢

时间:2023-07-29 11:00:16浏览次数:38  
标签:语句 刷脏页 innodb dirty SQL 脏页 执行 pages delete

一 现象

某个数据库经常在某个时间点比如凌晨2点或者白天某些时间段发出如下报警

  1. [Critical][prod][mysql] - 超200 kill SQL/分钟

  2. [P0][PROBLEM][all(#2) db_data.Com_kill db=XXXX[m]:3306 10.53333>=3.3]

  3. [O1 2019-11-01 03:40:00]

报警的意思是每分钟超过200个sql被kill,是一个严重告警级别,会打电话给DBA。大半夜报警的确令人不爽,那么如何解决这个问题呢?

通过检查日志,我们发现被kill的sql都是delete语句。业务方其实会定时的跑删除任务,这个任务涉及到N 多个表,删除任务持续时间比较长,所以白天和晚上都有一定概率会触发 sql-killer ,然后报警。

在有赞的数据库运维体系中,每个实例都会配置一个 sql-killer 的实时工具,用于kill query 超过指定阈值的sql请求(类似pt-killer)。

二 初步分析

在之前的案例分析过程中,碰到过因为长事务导致特定表上面的查询耗时增加的问题。经分析发现,这次被kill的SQL 是分布在各个表上面,而且查询发现并不存在长事务。

分析问题发生时候的数据库快照信息,QPS 都很低,除了差不多10 TPS 的DELETE和几十的SELECT,没有发现有问题的SQL。

分析当时的show engine innodb status 的信息,发现每次出问题的时候都会出现一些latch的等待,如下图所示。

图片

找到对应的代码行数

图片

看代码锁位置像是在等待各种Buffer Pool的各种latch。为啥会等待在这里呢,又没有DDL相关的SQL,于是百思不得其解。问题诊断一时间陷入困境。

三 抽丝剥茧

由于等待和Buffer Pool的各种latch相关,而且delete操作本身会产生大量脏数据,那会不会跟刷脏页操作相关呢?

我们看下SQL被kill的量和刷脏页的量之间的关系

 

 

发现每秒刷脏页的量和SQL被kill的量的曲线有点相近,看着刷脏页的量挺大的,但是每秒DELETE的TPS又不是很高,为啥这么低的TPS会让刷脏页频率抖动以及SQL执行变慢呢?

曾经换过不同批次的机器,发现问题依旧,并没有改善,说明并不是机器本身的问题。

继续浏览buffer pool相关的监控指标,像是发现新大陆一样的发现了一个异常指标

 

脏页比例达到了快90%!!!太吓人了!!!

为啥脏页比例会达到90%呢,无非就是刷脏页的速度跟不上产生的速度,要么就是IO能力不行,要么就是产生脏页的速度过快,要么就是内存池太小,导致Buffer Pool被脏页占满。

那么这个脏页比例达到快90% 会有什么问题呢?

MySQL有两个关于脏页的参数

  1. # yzsql 3306 param dirty

  2. Variable_name Value

  3. innodb_max_dirty_pages_pct 75.000000

  4. innodb_max_dirty_pages_pct_lwm 50.000000

我们查看下官方定义

innodb_max_dirty_pages_pct :

InnoDB tries to flush data from the buffer pool so that the percentage of dirty pages does not exceed this value. The default value is 75.

innodb_max_dirty_pages_pct 是为了避免脏页比例大于75%,改变该参数不会影响刷脏页的速度。

innodb_max_dirty_pages_pct_lwm:

Defines a low water mark representing the percentage of dirty pages at which preflushing is enabled to control the dirty page ratio. The default of 0 disables the pre-flushing behavior entirely.

innodb_max_dirty_pages_pct_lwm 表示的是当脏页比例达到该参数表示的低水位时候,刷脏线程就开始预刷脏来控制脏页比例,避免达到innodb_max_dirty_pages_pct 。刷脏页的最大IO能力是受innodb_io_capacity和 innodb_io_capacity_max 控制。

生产上我们将innodb_max_dirty_pages_pct_lwm设置成了50

当脏页比例大于 innodb_max_dirty_pages_pct 时候,InnoDB 会进行非常激烈的刷脏页操作,但是由于DELETE操作还是在进行,脏页产生的速度还是非常快,刷脏页的速度还是跟不上脏页产生的速度。为了避免脏页比例进一步扩大,更新将会被堵塞,从而导致DELETE 执行变慢,直至被KILL。

发现问题之后,根据我们之前的假设,有三种解决方案:

  1. 调大 io_capacity ,但是由于主机是多实例部署,IO占用已经比较高,PASS。

  2. 降低脏页产生速度,也就是调低DELETE速度,因为一天的速度产生很快,为了避免删除跟不上插入的速度,也被PASS

  3. 调大Buffer Pool,可以容纳更多的脏页。

说干就干,得益于MySQL 5.7的在线调整Buffer Pool,立马将Buffer Pool Size扩了一倍,效果非常显著

 

 

 

 

脏页比例立马下降,被kill的SQL也下降了。平均SQL rt下降很多。

四 总结

得益于MySQL的开源,很多错误都可以直接确认到对应的代码,大致定位到问题发生的地方,给问题排查带来了很多方便。同时对MySQL buffer pool 的命中率以及脏页比例也要多多关注,对SQL的性能都有很大的影响。

转载自爱可生公众号

标签:语句,刷脏页,innodb,dirty,SQL,脏页,执行,pages,delete
From: https://www.cnblogs.com/lovezhr/p/17589453.html

相关文章

  • 官网解决-找不到mfc140u.dll,无法继续执行代码
    问题:找不到mfc140u.dll,无法继续执行代码原因:一般都是电脑缺少或丢失了dll,或者没安装对应版本的vc运行库解决:直接去官网下载vc运行库,安装即可直接上Microsoft官网链接:https://answers.microsoft.com/zh-hans/windows/forum/all/%E5%BE%AE%E8%BD%AF%E5%B8%B8%E7%94%A8%E8%BF%90......
  • 2-9 在求对二的补码时,表达式 x &= (x – 1)可以删除 x 中最右边值为 1 的 一个二进制
    ArchlinuxGCC13.1.1 202304292023-07-2910:29:56星期六 点击查看代码#include<stdio.h>#include<stdint.h>intbitcount(uint8_tx){intcount=0;while(x!=0){x&=x-1;count++;}returncount;......
  • ajax动态加载JS不执行的解决办法
    //第一步:匹配加载的页面中是否含有jsvarregDetectJs=/<script(.|\n)*?>(.|\n|\r\n)*?<\/script>/ig;varjsContained=ajaxLoadedData.match(regDetectJs);//第二步:如果包含js,则一段一段的取出js再加载执行if(jsContained){ //分段取出js正则 varregGetJS=/<sc......
  • 利用pyinstaller发布不依赖python解释器的可执行exe
    现在打包app.py,从Windows命令提示符(cmd)运行:pyinstaller.exe--onefile--windowedapp.py就这么简单。如果打包成功,最终的可执行文件app.exe和任何相关文件将放在dist目录中,如果该目录不存在,将创建该目录。  PyInstallerManualVersionPyInstaller5.1Homepagehttps://pyinstall......
  • 命令执行_代码执行漏洞
    远程代码注入漏洞原理攻击者可利用代码注入漏洞执行任意代码,来操作服务器危害执行任意代码,来操作服务器操作数据库,插入恶意数据,可能获取系统权限攻击修改系统配置,修改网络配置,可能对服务器及网络造成影响可以进一步对网络渗透,由于代码注入攻击多半可获取系统权限,对网络的......
  • 数据库学习(一)——DDL数据库定义语句
    定义数据库创建数据库使用CRETEDATABASE关键字,指定编码和排序格式CREATEDATABASEmysqldbDEFAULTCHARACTERSETutf-8DEFAULTCOLLATEutf8_unicode_ci;使用IFNOTEXITS关键字CREATEDATABASEIFNOTEXITSmysqldb查看数据库查看数据库showdatabases;查......
  • 先谈流程,再论执行​
    没有完整的工作流程系统,谈执行力都是扯淡在公司的经营发展过程中,经常会出现以下不正常的情况,尤其是在中小民营企业里面:▼为什么下属自己的工作做完了,却不知道往下找谁呢?▼为什么下属做事拖拖拉拉呢?▼为什么一项工作领导等了很长时间却等不来结果呢?▼为什么在公司内部总是出现......
  • 基本的 HTTP 方法GET、POST、PUT 和 DELETE
    (目录)一、方法介绍1.1GETGET方法用于从服务器检索资源。这是一种安全的方法,因为它不会以任何方式改变资源的状态。GET方法是幂等的,因此多次调用此方法将始终给出相同的结果。HTTPGET'http://www.timi.com/users'HTTPGET'http://www.timi.com/users?size=20&page=5'HT......
  • 进程、线程与协程:并发执行的三种重要概念与应用
    在计算机科学中,进程、线程和协程是实现并发执行的三种重要概念。它们是计算机系统中的核心组成部分,为多任务处理和资源共享提供了有效的解决方案。本文将深入探讨进程、线程和协程的概念、特点及它们在并发编程和系统设计中的应用,帮助读者全面理解并发执行的原理和实践。进程的概念......
  • 写一条mysql更新数据的语句
    要更新MySQL数据库中的数据,你可以使用UPDATE语句。下面是一个示例代码,用于更新数据库中的数据:importmysql.connectordefupdate_data(connection,table,column,value,condition):cursor=connection.cursor()#构造更新数据的SQL语句update_query=f"UP......