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

一、背景

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索引,备份和还原

    1.索引  1.索引是占硬盘空间 ,也是按页存放的 。 思考题:一个索引页,(数据页)  占用多少个字节  。SQL Server 8192个字节   2.索引:是一种有效组合数据的方式,为了快速查找指定记录   3.唯一索引: 主键索引:如果一列是主键列,那么该列自动会提升成主键索引。   4.索引分类    普通索引    唯一索引    主键索引   …

    MySQL 2023年4月12日
    00
  • MySQL内联和外联查询

    内连: 内连接是通过在查询中设置连接条件的方式,来移除查询结果集中某些数据行后的交叉连接。简单来说,就是利用条件表达式来消除交叉连接的某些数据行。 在MySQL FROM 子句中使用关键字 INNER JOIN 连接两张表,并使用 ON 子句来设置连接条件。如果没有任何条件,INNER JOIN 和 CROSS JOIN 在语法上是等同的,两者可以互换。 语…

    MySQL 2023年4月13日
    00
  • Linux(CentOS7)安装与卸载MySQL8.0图文详解

    Mysql数据库的安装对于开发者来说,是我们必然会面对的问题,它的安装过程其实并不复杂,并且网络上的安装教程也非常多,但是对于新手来说,各种不同形式的安装教程,又给新手们带来了要选择哪种方式进行安装的难题,而且很多时候按照教程也没有能够安装成功,安装过程出现各种各样的错误。 下面记录了我在Linux(Centos 7)环境下安装Mysql的完整过程,实操记录…

    MySQL 2023年4月25日
    00
  • MySQL备份类型

    MySQL是一种用于管理数据的关系型数据库管理系统。MySQL备份是一种旨在保护数据库免遭数据丢失、损坏或被误删除等的操作,以便恢复数据库的数据的过程。MySQL备份有多种类型,包括物理备份、逻辑备份和增量备份。本文将详细介绍这三种类型。 物理备份 物理备份是备份数据库的一个镜像,包含所有数据和对象。它从硬盘级别上备份数据库,对所有表、数据和结构都会进行备份…

    MySQL 2023年3月10日
    00
  • 解决mysql的赋权操作之GRANT ALL PRIVILEGES ON *.* TO ‘root‘@‘%‘ IDENTIFIED BY ‘123456‘ WITH GRANT OPTION问题

    要解决MySQL的赋权操作,我们需要使用GRANT语句。具体流程如下: Step 1:登录MySQL 使用类似以下命令登录MySQL: mysql -u root -p 然后输入密码,按回车键。 Step 2:选择数据库 使用以下命令选择数据库: use mysql; 然后按回车键。 Step 3:创建用户并赋予权限 使用以下命令创建用户并赋予权限: GRA…

    MySQL 2023年5月18日
    00
  • c++连接mysql5.6的出错问题总结

    那我来为大家介绍如何解决“c++连接mysql5.6的出错问题总结”的完整攻略。 问题描述 在使用c++连接mysql 5.6的时候,有时会遇到一些常见的问题,如: mysql_library_init() 函数报错 mysql_init() 函数报错 执行mysql_real_connect() 函数报错 这些错误可能会导致连接mysql失败,影响程序的运…

    MySQL 2023年5月18日
    00
  • 一文带你了解MySQL四大类日志

    一下是“一文带你了解MySQL四大类日志”的完整攻略: 一文带你了解MySQL四大类日志 MySQL作为目前最流行的开源关系型数据库之一,拥有着丰富的特性和强大的功能。在它的运行过程中,MySQL会产生各种类型的日志,用于记录MySQL的运行状况和异常情况。MySQL日志主要可分为四大类:二进制日志、错误日志、查询日志和慢查询日志。 二进制日志 二进制日志(…

    MySQL 2023年5月18日
    00
  • 解决MySql客户端秒退问题(找不到my.ini)

    解决MySql客户端秒退问题(找不到my.ini) 问题描述 在启动MySql客户端时,出现了秒退现象,无论是通过命令行启动还是双击桌面图标启动,都会弹出一个命令行窗口短暂出现,然后又迅速消失。而当尝试通过cmd命令启动时,会提示找不到my.ini文件的错误信息。 问题原因 MySql客户端启动需要读取数据库的配置文件my.ini,如果MySql客户端找不到…

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