技术分享 | delete 语句引发大量 sql 被 kill 问题分析

技术分享 | delete 语句引发大量 sql 被 kill 问题分析作者:王航威 有赞 MySQL DBA,擅长分析和解决数据库的性能问题,利用自动化工具解决日常需求。 现象 某个数据库经常在某个时间点比如凌晨 2 点或者白天某些时间段发出如下报警 [Critica…

作者:王航威 有赞 MySQL DBA,擅长分析和解决数据库的性能问题,利用自动化工具解决日常需求。

现象

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

[Critical][prod][mysql] - 超200 kill SQL/分钟
[P0][PROBLEM][all(#2) db_data.Com_kill db=XXXX[m]:3306 10.53333>=3.3]
[O1 2019-11-01 03:40:00]

代码100分

报警的意思是每分钟超过 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 的等待,如下图所示。 技术分享 | delete 语句引发大量 sql 被 kill 问题分析

找到对应的代码行数 技术分享 | delete 语句引发大量 sql 被 kill 问题分析

看代码锁位置像是在等待各种 Buffer Pool 的各种 latch。为啥会等待在这里呢,又没有 DDL 相关的 SQL,于是百思不得其解。

问题诊断一时间陷入困境。

抽丝剥茧

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

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

技术分享 | delete 语句引发大量 sql 被 kill 问题分析 技术分享 | delete 语句引发大量 sql 被 kill 问题分析

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

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

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

技术分享 | delete 语句引发大量 sql 被 kill 问题分析

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

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

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

MySQL 有两个关于脏页的参数

代码100分# yzsql    3306 param  dirty
Variable_name	Value
innodb_max_dirty_pages_pct         75.000000
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 扩了一倍,效果非常显著

技术分享 | delete 语句引发大量 sql 被 kill 问题分析 技术分享 | delete 语句引发大量 sql 被 kill 问题分析 技术分享 | delete 语句引发大量 sql 被 kill 问题分析 技术分享 | delete 语句引发大量 sql 被 kill 问题分析

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

总结

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/10994.html

(0)
上一篇 2022-12-18
下一篇 2022-12-18

相关推荐

  • MySQL优化之索引解析[通俗易懂]

    MySQL优化之索引解析[通俗易懂]索引的本质 MySQL索引或者说其他关系型数据库的索引的本质就只有一句话,以空间换时间。 索引的作用 索引关系型数据库为了加速对表中行数据检索的(磁盘存储的)数据结构 索引的分类 数据结构上面的分类

    2023-05-09
    99
  • sqlserver索引优化及测试_mysql 索引优化

    sqlserver索引优化及测试_mysql 索引优化1.1、查找缺失索引 SELECT A.USER_SEEKS 查找次数,A.USER_SCANS 扫描次数, ROUND(A.AVG_TOTAL_USER_COST,2) 减少的用户查询的平均成本,A

    2023-04-23
    106
  • Python模块:Numpy计算数组

    Python模块:Numpy计算数组Numpy是Python中功能强大的库之一,它为Python提供了一种更高效的方法来处理大型数​​组和矩阵。Numpy使得Python变得更接近于一种与MATLAB和R等科学计算语言相似的语言,使其在科学计算和数据分析方面更有竞争力。它具有广泛的功能和易于使用的API,允许开发人员更快地进行数值计算。

    2024-03-06
    33
  • 解决Python中name ‘os’ is not defined错误

    解决Python中name ‘os’ is not defined错误在Python中,我们可以使用import语句来导入其他的Python模块,通过导入模块来调用其中定义的方法和变量,这也是Python强大的面向对象编程的基础之一。然而,当我们在程序中出现name ‘os’ is not defined错误时,就表示Python找不到名为os的模块或者变量。这种错误通常在导入模块时出现,而且它还说明了一个很重要的问题,就是Python的模块机制。

    2024-03-25
    32
  • python读取数据库

    python读取数据库一、access 1、获取表 http://blog.sina.com.cn/s/blog_92ab5a530102wjxe.html 参考这个链接。我把操作录制下来如下所示 import pypy…

    2023-03-24
    119
  • 数据库索引b树 b+树_b树索引

    数据库索引b树 b+树_b树索引一、什么是索引 在进行索引之前,我们先提前复习一下数据库的结构 数据库索引,是数据库管理系统中一个排序的数据结构,以协助快速查询、更新数据库表中数据。就像我们以前用的新华字典的目录一样,能帮助我们快…

    2023-03-26
    99
  • 5. 索引与算法—B+树的操作、辅助索引与聚集索引、Cardinality、联合索引、覆盖索引、MRR/ICP、哈希算法、全文索引「终于解决」

    5. 索引与算法—B+树的操作、辅助索引与聚集索引、Cardinality、联合索引、覆盖索引、MRR/ICP、哈希算法、全文索引「终于解决」5.3 B+ 树 B+ 树是为磁盘或其他直接存储辅助设备设计的一种平衡查找树。在B+树中,所有记录都是按照键值大小顺序存放在同一层的叶子节点上,由叶子节点指针进行连接,双向链表连接。 5.3.1 B+

    2023-02-18
    118
  • 包含python之jieba库使用的词条

    包含python之jieba库使用的词条自然语言处理(NLP)是机器学习重要分支之一,主要应用于篇章理解、文本摘要、情感分析、知识图谱、文本翻译等领域。而NLP应用首先是对文本进行分词,当前中文分词器有Ansj、paoding、盘古分词等多种,而最基础的分词器应该属于jieba分词器(比较见下图)。

    2023-11-27
    56

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注