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数据库,仅需这样一个shell脚本(推荐)

    这样一个shell脚本是指一个名为mysql_dropper.sh的脚本,它可以批量删除MySQL数据库。下面是详细的攻略: 下载和安装必要的工具 首先需要安装mysql命令行客户端和expect工具。在Linux下,可以通过以下命令安装: sudo apt-get update sudo apt-get install mysql-client expec…

    MySQL 2023年5月18日
    00
  • 这几个SQL语法的坑,你踩过吗

    本文已经收录到Github仓库,该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等核心知识点,欢迎star~ Github地址 大家好,我是大彬~ 今天给大家分享几个SQL常见的“坏毛病”及优化技巧。 SQL语句的执行顺…

    2023年4月8日
    00
  • MySQL问答系列之什么情况下会用到临时表

    MySQL问答系列之什么情况下会用到临时表 什么是临时表 临时表是一种特殊的表,用于存储临时数据。与普通表不同的是,临时表的数据只存在于当前会话中,一旦关闭该会话,临时表就会自动删除。 MySQL提供两种类型的临时表:局部临时表和全局临时表。局部临时表只能在当前数据库中使用,而全局临时表则可以在所有数据库中使用。 什么情况下会用到临时表 1. 临时存储查询结…

    MySQL 2023年5月19日
    00
  • 详解MySQL的约束

    在 MySQL 中,约束是指对表中数据的一种约束,能够帮助数据库管理员更好地管理数据库,并且能够确保数据库中数据的正确性和有效性。 例如,在数据表中存放年龄的值时,如果存入 200、300 这些无效的值就毫无意义了。因此,使用约束来限定表中的数据范围是很有必要的。 在 MySQL 中,支持以下几种约束: 1. 主键约束(Primary Key Constra…

    MySQL 2023年3月9日
    00
  • Centos 6.3将Mysql 5.1.61升级为mysql 5.6.19遇到的问题及解决方式

    更新yum源 CentOS 6.3自带的yum源中没有MySQL5.6,则需要添加新的yum源。 示例: sudo rpm -Uvh http://dev.mysql.com/get/mysql-community-release-el6-5.noarch.rpm 卸载旧版本Mysql 如果系统中已经安装了旧版本的MySQL,则需要卸载掉。 示例: sudo…

    MySQL 2023年5月18日
    00
  • MySQL中查询、删除重复记录的方法大全

    MySQL中查询、删除重复记录是使用非常频繁的操作,本文将详细讲解MySQL中查询、删除重复记录的方法大全。 1. 查询重复记录 1.1 利用GROUP BY和HAVING子句 SELECT column_name(s) FROM table_name GROUP BY column_name(s) HAVING COUNT(*) > 1; 上述代码中…

    MySQL 2023年5月19日
    00
  • MySQL explain 和 profiling 详解

    MySQL explain 和 profiling 详解 mysql explain MySQL 的 EXPLAIN 是一个用于查询优化的工具,它可以显示 MySQL 数据库如何执行查询。它返回一组关于查询执行计划的信息,包括用到的索引,表的连接顺序以及 MySQL 使用的查询类型。下面是 EXPLAIN 返回的列及其含义: id id:查询中每个 SELE…

    MySQL 2023年4月8日
    00
  • MySQL中常见的八种SQL错误用法示例

    MySQL中常见的八种SQL错误用法示例 本文介绍MySQL中八种常见的SQL错误用法,以及如何正确地使用它们。 把WHERE与LIMIT语句的顺序写反了 错误示例: SELECT * FROM users LIMIT 10 WHERE age > 18; 正确示例: SELECT * FROM users WHERE age > 18 LIMI…

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