MySQL:5.6 大事务show engine innodb status故障一例

MySQL:5.6 大事务show engine innodb status故障一例导读: 作者:高鹏(网名八怪),《深入理解MySQL主从原理32讲》系列文的作者。 今天遇到一个朋友的线上问题,大概意思就是说,我有一个线上的大事务大概100G左右,正在做回滚,当前看起来似乎影响了…

MySQL:5.6 大事务show engine innodb status故障一例

导读: 作者:高鹏(网名八怪),《深入理解MySQL主从原理32讲》系列文的作者。

今天遇到一个朋友的线上问题,大概意思就是说,我有一个线上的大事务大概100G左右,正在做回滚,当前看起来似乎影响了线上的业务,并且回滚很慢,是否可以减轻对线上业务的影响。并且朋友已经取消了双1设置,但是没有任何改观。版本MySQL 5.6首先我们需要知道的是,MySQL并不适合大事务,大概列举一些MySQL中大事务的影响:

  • binlog文件作为一次写入,会在sync阶段消耗大量的IO,会导致全库hang主,状态大多为query end。

  • 大事务会造成导致主从延迟。

  • 大事务可能导致某些需要备份挂起,原因在于flush table with read lock,拿不到MDL GLOBAL 级别的锁,等待状态为 Waiting for global read lock。

  • 大事务可能导致更大Innodb row锁加锁范围,导致row锁等待问题。

  • 回滚困难。

基于如上一些不完全的列举,我们应该在线上尽可能的避免大事务。好了我们下面来进行问题讨论。

一、问题

前面已经说了,我们已经取消了双1设置,所谓的双1就是 sync_binlog=1和 innodb_flush_log_at_trx_commit=1。这两个参数线上要保证为1,前者保证binlog的安全,后者保证redo的安全,它们在数据库crash recovery的时候起到了关键做用,不设置为双1可能导致数据丢失。具体的参数含义不做过多讨论。但是这里的问题是即便取消了双1,没有任何改观,因此似乎说明IO问题不是主要瓶颈呢?下面我们来看几个截图:

  • vmstat 截图

image

  • iostat 截图

image

image

  • top -Hu截图

image

我们重点观察vmstat的r 和 b列发现,IO队列没有有什么问题 并且wa%并不大。我们观察iostat中的%util和读写数据大小来看问题不大,并且tps远没达到极限(SSD盘)。我们top -Hu 可以观察到 %us不小,并且有线程已经打满了(99.4%CPU)一个CPU核。 因此我们可以将方向转为研究CPU瓶颈的产生,希望能够对问题有帮助,然后从提供的perf top中我们有如下发现:

image

好了我们将问题先锁定到lock_number_of_rows_locked这个函数上。

二、函数lock_number_of_rows_locked的作用

朋友用的5.6,但是我这里以5.7.26的版本进行描述。然后下一节描述5.6和5.7算法上的关键差异。不知道大家是否注意过show engine innodb status中的这样一个标志:

image

这个标记就来自函数lock_number_of_rows_locked,含义为当前事务加行锁的行数。而这个函数包裹在函数lock_print_info**_**all_transactions下面,lock_print_info_all_transactions函数是打印我们通常看到show engine innodb status中事务部分的核心参数。我们来看一下简单的流程:

PrintNotStarted print_not_started(file);//建立一个结构体,目的是做not start 事务的打印
ut_list_map(trx_sys->mysql_trx_list, print_not_started);  //这个地方打印出那些事务状态是no start的事务。mysql_trx_list是全事务。
const  trx_t* trx;
TrxListIterator trx_iter;  //这个迭代器是trx_sys->rw_trx_list 这个链表的迭代器
const  trx_t* prev_trx =  0;
/* Control whether a block should be fetched from the buffer pool. */
bool load_block =  true;
bool monitor = srv_print_innodb_lock_monitor &&  (srv_show_locks_held !=  0);
while  ((trx = trx_iter.current())  !=  0)  {  //通过迭代器进行迭代 ,显然这里不会有只读事务的信息,全部是读写事务。
   ...
    /* If we need to print the locked record contents then we
    need to fetch the containing block from the buffer pool. */
    if  (monitor)  {
         /* Print the locks owned by the current transaction. */
         TrxLockIterator& lock_iter = trx_iter.lock_iter();
         if  (!lock_trx_print_locks(  //打印出锁的详细信息
                  file, trx, lock_iter, load_block))

代码100分

简单的说就是先打印哪些处于not start的事务,然后打印那些读写事务的信息,当然我们的回滚事务肯定也包含在其中了,需要注意的是只读事务show engine不会打印。对于处于回滚状态的事务我们可以在show engine中观察到如下信息:

image

函数trx_print_low可以看到大部分的信息,这里就不详细解释了。既然如此我们需要明白lock_number_of_rows_locked是如何计算的,下面进行讨论。 三、函数lock_number_of_rows_locked的算法变化

上面我们说了函数lock_number_of_rows_locked函数会打印出当前事务加行锁的行数。那么我们来看一下5.6和5.7算法的不同。

  • 5.7.26

实际上只有如下一句话:

代码100分return(trx_lock->n_rec_locks);

我们可以看到这是返回了一个计数器,而这个计数器的递增就是在每行记录加锁后完成的,在函数lock_rec_set_nth_bit的末尾可以看到 ++lock->trx->lock.nreclocks ,因此这是一种预先计算的机制。因此这样的计算代价很低,也不会由于某个事务持有了大量的锁,而导致计算代价过高。

  • 5.6.22

随后我翻了一下5.6.22的代码,发现完全不同如下:

for  (lock  = UT_LIST_GET_FIRST(trx_lock->trx_locks);  //使用for循环每个获取的锁结构
lock  != NULL;
lock  = UT_LIST_GET_NEXT(trx_locks,  lock))  {
    if  (lock_get_type_low(lock)  == LOCK_REC)  {  //过滤为行锁
         ulint   n_bit;
         ulint   n_bits = lock_rec_get_n_bits(lock);
         for  (n_bit =  0; n_bit < n_bits; n_bit++)  {//开始循环每一个锁结构的每一个bit位进行统计
              if  (lock_rec_get_nth_bit(lock, n_bit))  {
                   n_records++;
              }
         }
    }
}
return(n_records);

我们知道循环本身是一种CPU密集型的操作,这里使用了嵌套循环实现。因此如果在5.6中如果出现大事务操作了大量的行,那么获取行锁记录的个数的时候,将会出现高耗CPU的情况。

四、原因总结和解决

有了上面的分析我们很清楚了,触发的原因有如下几点:

  • MySQL 5.6版本

  • 有大事务的存在,大概100G左右的数据加行锁了

  • 使用了show engine innodb status

这样当在统计这个大事务行锁个数的时候,就会进行大量的循环操作。从现象上看就是线程消耗了大量的CPU资源,并且处于perf top的第一位。

知道了原因就很简单了,找出为频繁使用show engine innodb status的监控工具,随后业务全部恢复正常,IO利用率也上升了如下:

image

当然如果能够使用更新的版本比如5.7及8.0 版本将不会出现这个问题,可以考虑使用更高版本。分析性能问题需要首先找到性能的瓶颈然后进行集中突破,比如本例中CPU资源消耗更加严重。也许解决问题就在一瞬间。

五、其他

最后通过朋友后面查询的bug如下:https://bugs.mysql.com/bug.php?id=68647发现印风(翟卫翔)已经在多年前提出过了这个问题,并且做出了修改意见,并且这个修改意见官方采纳了,也就是上面我们分析的算法改变。经过印风(翟卫翔)的测试有bug中有如下描述:

  • From perf top, function locknumberofrowslocked may occupy more than 20% of CPU sometimes

也就是CPU消耗会高达20%。

下面是5.7.26调用栈帧:

image

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

(0)
上一篇 2023-02-06
下一篇 2023-02-06

相关推荐

  • MySQL数据库(五)插入操作

    MySQL数据库(五)插入操作前提要述:参考书籍《MySQL必知必会》 《MySQL必知必会》是先讲了查询,但是没有记录就无法查询,所以先将如何添加数据。 表已经知道怎么创建了,随便创两张。 5.1 插入数据 MySQL使用 IN

    2023-01-22
    149
  • 伪分布式环境下启动Hadoop下的Hive[亲测有效]

    伪分布式环境下启动Hadoop下的Hive[亲测有效]在本地搭建好伪分布式环境,打开虚拟机进入linux系统,如果是在root用户下则需要切换至Hadoop用户 su – hadoop 按需输入hadoop密码 在hadoop家目录下启动Hadoop集群

    2023-03-07
    147
  • 提高代码效率的超实用小技巧,让你的Python sleep clock更智能化

    提高代码效率的超实用小技巧,让你的Python sleep clock更智能化a href=”https://www.python100.com/a/sm.html”font color=”red”免责声明/font/a a href=”https://beian.miit.gov.cn/”苏ICP备2023018380号-1/a Copyright www.python100.com .Some Rights Reserved.

    2024-01-31
    92
  • Python中使用元组进行不可变序列操作

    Python中使用元组进行不可变序列操作元组(Tuple)是Python中的一种不可变类型序列,用于存储一组数据。元组的创建方式与列表相似,用小括号 “( )” 将元素括起来,多个元素之间用逗号 “,” 隔开。虽然元组和列表都是序列类型,但元组不可变的特性使其在某些场景下具有优势,尤其是在保证数据不被修改的情况下,可以提高代码的安全性和效率。

    2023-12-06
    105
  • excel怎么截取字符串中一段_substring截取字符串

    excel怎么截取字符串中一段_substring截取字符串excel中截取字符串最长用到的方法是函数法,常用的字符串截取函数包括left,right,mid函数等。本文首先解释上述三个常用函数的语法,然

    2023-03-01
    127
  • 怎么安装phpMyAdmin?

    怎么安装phpMyAdmin?安装phpmyadmin的步骤:1、到phpMyAdmin官方网站下载,再解压到web可以访问的目录下;2、打开libraries目录下的config.default.php文件进行配置(位置访问网…

    2022-12-20
    149
  • Redis设计与实现2.1:主从复制[通俗易懂]

    Redis设计与实现2.1:主从复制[通俗易懂]主从复制 这是《Redis设计与实现》系列的文章,系列导航:Redis设计与实现笔记 SLAVEOF 新旧复制功能 旧版复制功能 旧版复制功能的实现为 同步 和 命令传播: 当刚连上Master时,要

    2023-05-14
    148
  • MySQL深入学习-

    MySQL深入学习-B+树索引的正确使用 索引并不是越多越好,索引创建越多,MySQL维护的代价越高,如果SQL未能完全使用到索引,创建索引的意义是不大的。 适用条件 表x,创建索引a,b,c。主键y。 全值匹配 sel

    2023-05-18
    140

发表回复

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