Mysql Online DDL的使用详解


Posted in MySQL onMay 20, 2021

正文

Online DDL在MySQL 5.6才开始支持的,在5.5及之前版本,使用alter table/create index等命令进行表结构修改操作均会锁表,这在生产环境上明显是不可接受的。

在MySQL 5.7,Online DDL在性能和稳定性上不断得到优化,性能有显著优势,且对业务负载影响小,停写时间可控,相对pt-osc/gh-ost来说,无需安装第三方依赖包,同时支持Inplace算法的Online DDL,由于无需拷表,所需磁盘空间也更小。

先来看一个常见的DDL语句:

ALTER TABLE tbl_name ADD PRIMARY KEY (column), ALGORITHM=INPLACE, LOCK=NONE;

其中,LOCK描述了DDL期间运行的并发程度,ALGORITHM描述了DDL的实现方式

LOCK参数

  1. LOCK=NONE:允许并发的查询和DML操作
  2. LOCK=SHARED:允许并发的查询,但阻塞DML操作
  3. LOCK=DEFAULT: 由系统决定,允许尽可能多的并发性(并发查询、DML或两者)。如果省略LOCK子句相当于指定LOCK=DEFAULT
  4. LOCK=EXCLUSIVE:阻塞并发查询和DML操作。

ALGORITHM参数

  1. ALGORITHM=COPY:采用拷表方式进行表变更,与pt-osc/gh-ost类似;
  2. ALGORITHM=INPLACE:仅需要进行引擎层数据改动,不涉及Server层;

COPY TABLE流程

  1. 首先建立临时表,表结构为ALTAR TABLE更改后的结构
  2. 将原表中数据导入到临时表(server层创建临时表,会有显示的IBD文件)
  3. 删除原表
  4. 将临时表rename为原来的表名

同时这一过程中,为了保持数据的一致性,中间复制数据时(Copy Table)全程锁表只读,如果有写请求进来将无法提供服务,将导致连接数爆张。

IN-PLACE流程

  1. 建立一个临时文件,扫描原表主键的所有数据页
  2. 用数据页中原表记录生成B+树,存储到临时文件中(innodb_temp_data_file_path临时表空间下创建临时文件)
  3. 生成临时文件的过程中,将所有对原表的操作记在一个日志文件(rowlog)中
  4. 临时文件生成后,将日志文件中的操作应用到临时文件,得到一个辑数据上与原表相同
  5. 数据文件(日志文件记录和重放操作)
  6. 用临时文件替换原表数据文件

这一过程中,alter 语句在启动的时候获取MDL写锁,但是这个写锁在真正拷贝数据之前就退化成读锁,也就是说在最耗时的copy数据到临时文件的过程中,原表是可以进行dml操作的,仅仅会在最后的新旧表切换阶段加锁,这个rename的时间就非常快了。

允许并发DML的DDL操作

  • 创建/新增二级索引
  • 重命名二级索引
  • 删除二级索引
  • 改变索引类型(USING {BTREE | HASH})
  • 添加主键(expensive cost)
  • 删除主键并增加另一个(expensive cost)(ALTER TABLE tbl_name DROP PRIMARY KEY, ADD PRIMARY KEY (column), ALGORITHM=INPLACE, LOCK=NONE;)
  • 新增列 (expensive cost)
  • 删除列 (expensive cost)
  • 重命名列
  • 列重新排序 (expensive cost)
  • 改变列默认值
  • 删除列默认值
  • 改变列自增值
  • 设置列属性null/not null (expensive cost)
  • 修改枚举或集合列的定义
  • Change ROW_FORMAT
  • Change key block size

标记为expensive cost的操作虽然允许OnlineDDL,但本身对服务器IO,CPU都会造成较高负担,同时会导致复制阻塞,造成另一种形式的从库复制延迟,所以如果是大表,建议业务低峰期执行

不允许并发DML的DDL操作

  • 添加全文索引
  • 添加空间索引
  • 删除主键
  • 改变列数据类型
  • 添加自增列(新增列->变为自增列)
  • 变更表字符集
  • 修改数据类型长度
    • 特例:varchar字符长度从10变更到小于255 采用inplace方式不会锁表;从255变更到10会锁表;

以上就是Mysql Online DDL的使用详解的详细内容,更多关于Mysql Online DDL的使用的资料请关注三水点靠木其它相关文章!

MySQL 相关文章推荐
教你用eclipse连接mysql数据库
Apr 22 MySQL
MySQL查询学习之基础查询操作
May 08 MySQL
Mysql基础知识点汇总
May 26 MySQL
Mysql数据库索引面试题(程序员基础技能)
May 31 MySQL
mysql中between的边界,范围说明
Jun 08 MySQL
Unity连接MySQL并读取表格数据的实现代码
Jun 20 MySQL
SQL实现LeetCode(177.第N高薪水)
Aug 04 MySQL
MySQL中连接查询和子查询的问题
Sep 04 MySQL
MySQL面试题讲解之如何设置Hash索引
Nov 01 MySQL
面试提问mysql一张表到底能存多少数据
Mar 13 MySQL
MySQL 主从复制数据不一致的解决方法
Mar 18 MySQL
MySQL示例讲解数据库约束以及表的设计
Jun 16 MySQL
MySQL 存储过程的优缺点分析
May 20 #MySQL
IDEA 链接Mysql数据库并执行查询操作的完整代码
MySQL 覆盖索引的优点
May 19 #MySQL
MySQL 视图(View)原理解析
超详细教你怎么升级Mysql的版本
详解mysql三值逻辑与NULL
MySQL时间盲注的五种延时方法实现
You might like
为PHP初学者的8点有效建议
2010/11/20 PHP
PHP乱码问题,UTF-8乱码常见问题小结
2012/04/09 PHP
使用PHP编写的SVN类
2013/07/18 PHP
什么是OneThink oneThink后台添加插件步骤
2016/04/13 PHP
Yii框架分页实现方法详解
2017/05/20 PHP
javascript 弹出窗口中是否显示地址栏的实现代码
2011/04/14 Javascript
Dom 学习总结以及实例的使用介绍
2013/04/24 Javascript
利用JQuery和Servlet实现跨域提交请求示例分享
2014/02/12 Javascript
jQuery实现信息提示框(带有圆角框与动画)效果
2015/08/07 Javascript
分分钟玩转Vue.js组件(二)
2017/03/01 Javascript
EasyUI为Numberbox添加blur事件的方法
2017/03/05 Javascript
node.js中express-session配置项详解
2017/05/31 Javascript
JS获取填报扩展单元格控件的值的解决办法
2017/07/14 Javascript
Vue组件选项props实例详解
2017/08/18 Javascript
Vue中this.$router.push参数获取方法
2018/02/27 Javascript
vue代理和跨域问题的解决
2018/07/18 Javascript
express中static中间件的具体使用方法
2019/10/17 Javascript
原生javascript如何实现共享onload事件
2020/07/03 Javascript
[01:09:19]DOTA2-DPC中国联赛 正赛 VG vs Aster BO3 第二场 2月28日
2021/03/11 DOTA
PyQt5每天必学之带有标签的复选框
2018/04/19 Python
Python 使用PIL numpy 实现拼接图片的示例
2018/05/08 Python
python处理csv中的空值方法
2018/06/22 Python
python 平衡二叉树实现代码示例
2018/07/07 Python
pandas ix &iloc &loc的区别
2019/01/10 Python
​如何愉快地迁移到 Python 3
2019/04/28 Python
Python插件机制实现详解
2020/05/04 Python
Pytorch 使用CNN图像分类的实现
2020/06/16 Python
使用PyCharm官方中文语言包汉化PyCharm
2020/11/18 Python
Booking.com荷兰:全球酒店网上预订
2017/08/22 全球购物
关于圣诞节的广播稿
2014/01/26 职场文书
优秀的导游求职信范文
2014/04/06 职场文书
关于工作经历的证明书
2014/10/11 职场文书
关爱留守儿童捐款倡议书
2015/04/27 职场文书
惊涛骇浪观后感
2015/06/05 职场文书
少儿励志名言(80句)
2019/08/14 职场文书
关于Python OS模块常用文件/目录函数详解
2021/07/01 Python