MySQL8.0中Online DDL也要在业务低峰期执行

yizhihongxing

一、背景

MySQL5.6开始引入了Online DDLalter操作不再阻塞dml。在MySQL 8.0中,针对Online DDL做了进一步优化,alter table加列操作支持INSTANT算法,意思就是使用这个算法进行加列操作只需要修改表的元数据信息,操作瞬间就完成了。在MySQL 8.0.30以后,instant算法支持加列加到表的任一位置,并且也支持删列、重命名表等DDL操作。实际DDL中支持Online DDL的操作默认都会使用 ALGORITHM=INSTANT

二、问题

那么既然现在MySQLDDL这么快,我们是不是随便什么时候都可以去数据库中对表进行DDL呢?其实不是的,即使是Online DDL也要在业务低峰期进行。如果在对表进行Online DDL的时候刚好这个表有个慢查询在执行,那么DDL语句将等待这个查询的元数据锁(metadata_lock),后续对这个表的所有DML语句都将被这个DDL阻塞,进而很容易造成连环堵塞和CPU飙升的状况,对业务系统产生极大的影响。

三、实验

下面,将实际演示一下Online DDL引发的阻塞问题:

会话A:开启一个事务,执行一条select不提交,那这个事务将一直持有表notest的元数据锁。

 

mysql> begin;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test.notest where id=1;
Empty set (0.00 sec)

 

会话B:对表进行加列的DDL操作,可以看到该操作被堵塞,其实就是在等待会话A的元数据锁。

 

mysql> alter table test.notest add age int;

 

会话C、会话D:再开启两个会话,对该表进行简单的查询,该查询也在等待锁,没有返回结果。

 

mysql> select * from test.notest;

 

查看processlist,可以看到ddldml都在等待表的元数据锁:

 

mysql> show processlist;
+-------+-----------------+--------------------+------+------------------+-------+-----------------------------------------------------------------+-------------------------------------+
| Id    | User            | Host               | db   | Command          | Time  | State                                                           | Info                                |
+-------+-----------------+--------------------+------+------------------+-------+-----------------------------------------------------------------+-------------------------------------+
|     5 | event_scheduler | localhost          | NULL | Daemon           | 15986 | Waiting on empty queue                                          | NULL                                |
|    33 | repl            | 10.2.111.193:33644 | NULL | Binlog Dump GTID | 15964 | Source has sent all binlog to replica; waiting for more updates | NULL                                |
| 17805 | root            | localhost          | test | Query            |    64 | Waiting for table metadata lock                                 | alter table test.notest add age int |
| 17814 | root            | localhost          | test | Sleep            |   346 |                                                                 | NULL                                |
| 17973 | root            | localhost          | NULL | Sleep            |   368 |                                                                 | NULL                                |
| 18370 | root            | localhost          | NULL | Query            |    42 | Waiting for table metadata lock                                 | select * from test.notest           |
| 18393 | root            | localhost          | NULL | Query            |    24 | Waiting for table metadata lock                                 | select * from test.notest           |
| 18418 | root            | localhost          | NULL | Query            |     0 | init                                                            | show processlist                    |
+-------+-----------------+--------------------+------+------------------+-------+-----------------------------------------------------------------+-------------------------------------+
8 rows in set (0.00 sec)

 

查看元数据锁监控表performance_schema.metadata_locks表信息,可以看到当前数据库中存在的元数据锁已经元数据锁的对象和锁类型(在MySQL中,为了提高数据库的并发度,元数据锁被细分为了11种类型)。可以看到DDL语句给表带来的元数据锁类型为EXCLUSIVE,元数据EXCLUSIVE锁被持有期间任何其他的元数据锁都不能被授予,所以就阻塞了后续对表的所有DML操作,也包括select

 

mysql> select * from performance_schema.metadata_locks;
+-------------+--------------------+----------------+-------------+-----------------------+---------------------+---------------+-------------+--------------------+-----------------+----------------+
| OBJECT_TYPE | OBJECT_SCHEMA      | OBJECT_NAME    | COLUMN_NAME | OBJECT_INSTANCE_BEGIN | LOCK_TYPE           | LOCK_DURATION | LOCK_STATUS | SOURCE             | OWNER_THREAD_ID | OWNER_EVENT_ID |
+-------------+--------------------+----------------+-------------+-----------------------+---------------------+---------------+-------------+--------------------+-----------------+----------------+
| TABLE       | test               | notest         | NULL        |       140139226332208 | SHARED_READ         | TRANSACTION   | GRANTED     | sql_parse.cc:6085  |           18049 |              5 |
| GLOBAL      | NULL               | NULL           | NULL        |             132890928 | INTENTION_EXCLUSIVE | STATEMENT     | GRANTED     | sql_base.cc:5476   |           17881 |             21 |
| BACKUP LOCK | NULL               | NULL           | NULL        |             123469776 | INTENTION_EXCLUSIVE | TRANSACTION   | GRANTED     | sql_base.cc:5483   |           17881 |             21 |
| SCHEMA      | test               | NULL           | NULL        |             125839424 | INTENTION_EXCLUSIVE | TRANSACTION   | GRANTED     | sql_base.cc:5463   |           17881 |             21 |
| TABLE       | test               | notest         | NULL        |             125839520 | SHARED_UPGRADABLE   | TRANSACTION   | GRANTED     | sql_parse.cc:6085  |           17881 |             21 |
| TABLESPACE  | NULL               | test/notest    | NULL        |             130194048 | INTENTION_EXCLUSIVE | TRANSACTION   | GRANTED     | lock.cc:808        |           17881 |             21 |
| TABLE       | test               | #sql-5246_458d | NULL        |             124845680 | EXCLUSIVE           | STATEMENT     | GRANTED     | sql_table.cc:17024 |           17881 |             21 |
| TABLE       | test               | notest         | NULL        |             126124176 | EXCLUSIVE           | TRANSACTION   | PENDING     | mdl.cc:3754        |           17881 |             22 |
| TABLE       | test               | notest         | NULL        |       140138743169920 | SHARED_READ         | TRANSACTION   | PENDING     | sql_parse.cc:6085  |           18446 |              3 |
| TABLE       | performance_schema | metadata_locks | NULL        |       140138825181536 | SHARED_READ         | TRANSACTION   | GRANTED     | sql_parse.cc:6085  |           17890 |             26 |
| TABLE       | test               | notest         | NULL        |       140139414229984 | SHARED_READ         | TRANSACTION   | PENDING     | sql_parse.cc:6085  |           18469 |              3 |
+-------------+--------------------+----------------+-------------+-----------------------+---------------------+---------------+-------------+--------------------+-----------------+----------------+
11 rows in set (0.00 sec)

 

同时,可以查询sys.schema_table_lock_waits视图,该视图显示了当前元数据锁的锁等待信息。可以看到等待和阻塞的会话ID,并且kill掉阻塞会话的语句也直接在sql_kill_blocking_connection这一列给出来了,非常方便。

 

mysql> select * from sys.schema_table_lock_waits;
+---------------+-------------+-------------------+-------------+-----------------+-------------------+-----------------------+-------------------------------------+--------------------+-----------------------------+-----------------------------+--------------------+--------------+------------------+--------------------+------------------------+-------------------------+------------------------------+
| object_schema | object_name | waiting_thread_id | waiting_pid | waiting_account | waiting_lock_type | waiting_lock_duration | waiting_query                       | waiting_query_secs | waiting_query_rows_affected | waiting_query_rows_examined | blocking_thread_id | blocking_pid | blocking_account | blocking_lock_type | blocking_lock_duration | sql_kill_blocking_query | sql_kill_blocking_connection |
+---------------+-------------+-------------------+-------------+-----------------+-------------------+-----------------------+-------------------------------------+--------------------+-----------------------------+-----------------------------+--------------------+--------------+------------------+--------------------+------------------------+-------------------------+------------------------------+
| test          | notest      |             17881 |       17805 | root@localhost  | EXCLUSIVE         | TRANSACTION           | alter table test.notest add age int |                 53 |                           0 |                           0 |              18049 |        17973 | root@localhost   | SHARED_READ        | TRANSACTION            | KILL QUERY 17973        | KILL 17973                   |
| test          | notest      |             18446 |       18370 | root@localhost  | SHARED_READ       | TRANSACTION           | select * from test.notest           |                 47 |                           0 |                           0 |              18049 |        17973 | root@localhost   | SHARED_READ        | TRANSACTION            | KILL QUERY 17973        | KILL 17973                   |
| test          | notest      |             18469 |       18393 | root@localhost  | SHARED_READ       | TRANSACTION           | select * from test.notest           |                 44 |                           0 |                           0 |              18049 |        17973 | root@localhost   | SHARED_READ        | TRANSACTION            | KILL QUERY 17973        | KILL 17973                   |
| test          | notest      |             17881 |       17805 | root@localhost  | EXCLUSIVE         | TRANSACTION           | alter table test.notest add age int |                 53 |                           0 |                           0 |              17881 |        17805 | root@localhost   | SHARED_UPGRADABLE  | TRANSACTION            | KILL QUERY 17805        | KILL 17805                   |
| test          | notest      |             18446 |       18370 | root@localhost  | SHARED_READ       | TRANSACTION           | select * from test.notest           |                 47 |                           0 |                           0 |              17881 |        17805 | root@localhost   | SHARED_UPGRADABLE  | TRANSACTION            | KILL QUERY 17805        | KILL 17805                   |
| test          | notest      |             18469 |       18393 | root@localhost  | SHARED_READ       | TRANSACTION           | select * from test.notest           |                 44 |                           0 |                           0 |              17881 |        17805 | root@localhost   | SHARED_UPGRADABLE  | TRANSACTION            | KILL QUERY 17805        | KILL 17805                   |
+---------------+-------------+-------------------+-------------+-----------------+-------------------+-----------------------+-------------------------------------+--------------------+-----------------------------+-----------------------------+--------------------+--------------+------------------+--------------------+------------------------+-------------------------+------------------------------+
6 rows in set (0.01 sec)

 

四、结论

通过以上的实验,可以得出结论,如果在对表进行Online DDL时,该表上存在元数据锁,那么DDL将一直等待元数据锁释放,直到超过参数lock_wait_timeout的超时时间,并且该DDL会阻塞后续对该表的所有操作。因此,即使现在支持instant算法的Online DDL可以秒加列,也要在业务低峰期进行,并且执行DDL前,最好查询一下performance_schema.metadata_locks,检查是否存在元数据锁。

 

原文链接:https://www.cnblogs.com/coygfly/p/17351903.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:MySQL8.0中Online DDL也要在业务低峰期执行 - Python技术站

(0)
上一篇 2023年4月25日
下一篇 2023年4月25日

相关文章

  • MySQL系列详解五: xtrabackup实现完全备份及增量备份详解-技术流ken

      xtrabackup简介   xtrabackup是一个用来对mysql做备份的工具,它可以对innodb引擎的数据库做热备。xtrabackup备份和还原速度快,备份操作不会中断正在执行的事务,备份完成之后可以自动做校验,备份结果可以压缩(节省磁盘和带宽)。实际工作中可以用来做mysql的完全备份,增量备份,以及差异备份等。 xtrabackup有两个…

    MySQL 2023年4月13日
    00
  • mysql innodb 异常修复经验分享

    MySQL InnoDB 异常修复经验分享 背景 MySQL作为开源社区最常用的关系型数据库之一,广泛应用于互联网行业。但是,InnoDB引擎下的MySQL还是存在一些异常情况,例如崩溃、误删等,这些异常往往会导致数据丢失和业务中断。因此,这篇文章将分享在修复MySQL InnoDB异常的过程中所需要的经验和方法。 注意事项 在修复MySQL InnoDB异…

    MySQL 2023年5月18日
    00
  • Mysql报错Duplicate entry ‘值’ for key ‘字段名’的解决方法

    下面是详细讲解: 1. 什么是”Duplicate entry ‘值’ for key ‘字段名'”错误? “Duplicate entry ‘值’ for key ‘字段名'”即为MySQL的一个报错,意为”字段名”的值出现了重复。这个错误通常是由于对数据库进行插入或更新数据时,数据库已经存在相同的数据导致的。 2. “Duplicate entry ‘值…

    MySQL 2023年5月18日
    00
  • MySQL转义字符的使用方法

    MySQL转义字符是一些特殊字符,用于告诉MySQL将其视为普通字符,而不是语句的一部分。常见的转义字符包括反斜杠“\”、“单引号” ‘ ’、“双引号” " "、“换行符” \n、“制表符” \t等。以下是MySQL转义字符的使用方法及实例说明。 使用反斜杠转义特殊字符 反斜杠是MySQL中最常用的转义字符。它可以转义各种特殊字符,如单引…

    MySQL 2023年3月9日
    00
  • 拒绝“爆雷”!GaussDB(for MySQL)新上线了这个功能

    摘要:智能把控大数据量查询,防患系统奔溃于未然。 本文分享自华为云社区《拒绝“爆雷”!GaussDB(for MySQL)新上线了这个功能》,作者:GaussDB 数据库。 什么是最大读取行 一直以来,大数据量查询是数据库DBA们调优的重点,DBA们通常十八般武艺轮番上阵以期提升大数据查询的性能:例如分库分表、给表增加索引、设定合理的WHERE查询条件、限定…

    2023年4月8日
    00
  • 解决MySQL数据库链接超时报1129错误问题

    接下来我将详细讲解“解决MySQL数据库链接超时报1129错误问题”的完整攻略,过程中会示范两条具体的解决方案。 解决MySQL数据库链接超时报1129错误问题的完整攻略 问题描述 在使用MySQL数据库时,经常会遇到如下错误信息: ERROR 1129 (HY000): Host ‘xxx.xxx.xxx.xxx’ is blocked because o…

    MySQL 2023年5月18日
    00
  • MySQL 8.0数据字典有什么变化

    GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源。 GreatSQL是MySQL的国产分支版本,使用上与MySQL一致。 作者: 叶金荣 文章来源:GreatSQL社区原创 1. MySQL 8.0数据字典有什么变化 从MySQL 8.0开始,采用独立表空间模式的每个InnoDB表只有一个 .ibd 表空间文件,而不再有 .frm …

    MySQL 2023年4月18日
    00
  • MySQL修改密码的3种方式

    MySQL是一款开源的关系型数据库管理系统,被广泛应用于各行各业。为了保证数据库的安全,我们需要定期修改数据库的密码。本文将介绍MySQL修改密码的三种方式,包括使用命令行修改密码、使用MySQL Workbench修改密码、以及重置MySQL root密码。 使用命令行修改密码 1 登录MySQL 打开命令行工具,输入以下命令登录MySQL: mysql …

    MySQL 2023年3月10日
    00
合作推广
合作推广
分享本页
返回顶部