本文共 9444 字,大约阅读时间需要 31 分钟。
如果正在看这篇文章,相信你已经知道自己的需求了。
在 mysql 5.5 版本以前,修改表结构如添加索引、修改列,需要锁表,期间不能写入,对于大表这简直是灾难。从5.5特别是5.6里,情况有了好转,支持Online DDL,相关介绍见 ,而我在实际alter table过程中还是会引起 data meta lock 问题。pt-online-schema-change是Percona-toolkit一员,通过改进原生ddl的方式,达到不锁表在线修改表结构。
只介绍部分常用的选项
--host=xxx --user=xxx --password=xxx
连接实例信息,缩写-h xxx -u xxx -p xxx
,密码可以使用参数--ask-pass
手动输入。--alter
ALTER TABLE
关键字。与原始ddl一样可以指定多个更改,用逗号分隔。 change col1 col1_new type constraint
(保持类型和约束一致,否则相当于修改 column type,不能online)--alter "DROP FOREIGN KEY _fk_foo"
D=db_name,t=table_name
指定要ddl的数据库名和表名--max-load
默认为Threads_running=25
。每个chunk拷贝完后,会检查 SHOW GLOBAL STATUS 的内容,检查指标是否超过了指定的阈值。如果超过,则先暂停。这里可以用逗号分隔,指定多个条件,每个条件格式: status指标=MAX_VALUE
或者status指标:MAX_VALUE
。如果不指定MAX_VALUE,那么工具会这只其为当前值的120%。因为拷贝行有可能会给部分行上锁,Threads_running 是判断当前数据库负载的绝佳指标。
--max-lag
默认1s。每个chunk拷贝完成后,会查看所有复制Slave的延迟情况(Seconds_Behind_Master
)。要是延迟大于该值,则暂停复制数据,直到所有从的滞后小于这个值。--check-interval
配合使用,指定出现从库滞后超过 max-lag,则该工具将睡眠多长时间,默认1s,再检查。如--max-lag=5 --check-interval=2
。熟悉percona-toolkit的人都知道--recursion-method
可以用来指定从库dsn记录。另外,如果从库被停止,将会永远等待,直到从开始同步,并且延迟小于该值。
--chunk-time
默认0.5s,即拷贝数据行的时候,为了尽量保证0.5s内拷完一个chunk,动态调整chunk-size的大小,以适应服务器性能的变化。也可以通过另外一个选项--chunk-size
禁止动态调整,即每次固定拷贝 1k 行,如果指定则默认1000行,且比 chunk-time 优先生效
--set-vars
使用pt-osc进行ddl要开一个session去操作,set-vars
可以在执行alter之前设定这些变量,比如默认会设置--set-vars "wait_timeout=10000,innodb_lock_wait_timeout=1,lock_wait_timeout=60"
。因为使用pt-osc之后ddl的速度会变慢,所以预计2.5h只能还不能改完,记得加大wait_timeout
。
--dry-run
创建和修改新表,但不会创建触发器、复制数据、和替换原表。并不真正执行,可以看到生成的执行语句,了解其执行步骤与细节,和--print
配合最佳。。--execute
确定修改表,则指定该参数。真正执行alter。--dry-run与--execute必须指定一个,二者相互排斥这个很容易理解,pt-osc会在原表上创建3个触发器,而一个表上不能同时有2个相同类型的触发器,为简单通用起见,只能一棍子打死。
所以如果要让它支持有触发器存在的表也是可以实现的,思路就是:先找到原表触发器定义;重写原表触发器;最后阶段将原表触发器定义应用到新表。这其实是我的一个顾虑,因为如果update t1,触发update t2,但这条数据还没copy到t2,不就有异常了吗?后台通过打开general_log,看到它创建的触发器:
6165 Query CREATE TRIGGER `pt_osc_confluence_sbtest3_del` AFTER DELETE ON `confluence`.`sbtest3` FOR EACH ROW DELETE IGNORE FROM `confluence`.`_sbtest3_new` WHERE `confluence`.`_sbtest3_new`.`id` <=> OLD.`id` 6165 Query CREATE TRIGGER `pt_osc_confluence_sbtest3_upd` AFTER UPDATE ON `confluence`.`sbtest3` FOR EACH ROW REPLACE INTO `confluence`.`_sbtest3_new` (`id`, `k`, `c`, `pad`) VALUES (NEW.`id`, NEW.`k`, NEW.`c`, NEW.`pad`) 6165 Query CREATE TRIGGER `pt_osc_confluence_sbtest3_ins` AFTER INSERT ON `confluence`.`sbtest3` FOR EACH ROW REPLACE INTO `confluence`.`_sbtest3_new` (`id`, `k`, `c`, `pad`) VALUES (NEW.`id`, NEW.`k`, NEW.`c`, NEW.`pad`)并且copy操作是: 6165 Query INSERT LOW_PRIORITY IGNORE INTO `confluence`.`_sbtest3_new` (`id`, `k`, `c`, `pad`) SELECT `id`, `k`, `c`, `pad` FROM `confluence`.`sbtest3` FORCE INDEX(`PRIMARY`) WHERE ((`id` >= '4692805')) AND ((`id` <= '4718680')) LOCK IN SHARE MODE /*pt-online-schema-change 46459 copy nibble*/
在原表上update,新临时表上是replace into整行数据,所以达到有则更新,无则插入。同时配合后面的 insert ignore,保证这条数据不会因为重复而失败。
假设 t1 是要修改的表,t2 有外键依赖于 t1,_t1_new 是 alter t1 产生的新临时表。
这里的外键不是看t1上是否存在外键,而是作为子表的 t2。主要问题在 rename t1 时,t1“不存在”导致t2的外键认为参考失败,不允许rename。pt-osc提供--alter-foreign-keys-method
选项来决定怎么处理这种情况: rebuild_constraints
,优先采用这种方式
涉及的主要方法在 pt-online-schema-change
文件的 determine_alter_fk_method, rebuild_constraints, swap_tables三个函数中。
drop_swap
,
FOREIGN_KEY_CHECKS=0
我们的开发规范决定,即使表间存在外键参考关系,也不通过表定义强制约束。
使用OSC会使增加一倍的空间,包括索引
而且在 Row Based Replication 下,还会写一份binlog。不要想当然使用--set-vars
去设置 sql_log_bin=0,因为在这个session级别,alter语句也要在从库上执行,除非你对从库另有打算。 借助percona博客一张图说明一下:
[root@ssd-34 sysbench]# pt-online-schema-change --user=user --password=password --host=10.0.201.34 --alter "ADD COLUMN f_id int default 0" D=confluence,t=sbtest3 --print --executeNo slaves found. See --recursion-method if host ssd-34 has slaves.Not checking slave lag because no slaves were found and --check-slave-lag was not specified.Operation, tries, wait: analyze_table, 10, 1 copy_rows, 10, 0.25 create_triggers, 10, 1 drop_triggers, 10, 1 swap_tables, 10, 1 update_foreign_keys, 10, 1Altering `confluence`.`sbtest3`...Creating new table... ==> 创建新表CREATE TABLE `confluence`.`_sbtest3_new` ( `id` int(10) unsigned NOT NULL AUTO_INCREMENT, `k` int(10) unsigned NOT NULL DEFAULT '0', `c` char(120) COLLATE utf8_bin NOT NULL DEFAULT '', `pad` char(60) COLLATE utf8_bin NOT NULL DEFAULT '', PRIMARY KEY (`id`), KEY `k_3` (`k`)) ENGINE=InnoDB AUTO_INCREMENT=5000001 DEFAULT CHARSET=utf8 COLLATE=utf8_bin MAX_ROWS=1000000Created new table confluence._sbtest3_new OK.Altering new table... ==> 使用ddl修改新表结构ALTER TABLE `confluence`.`_sbtest3_new` ADD COLUMN f_id int default 0Altered `confluence`.`_sbtest3_new` OK.2016-05-24T20:54:23 Creating triggers... ==> 在旧表上创建3个触发器CREATE TRIGGER `pt_osc_confluence_sbtest3_del` AFTER DELETE ON `confluence`.`sbtest3` FOR EACH ROW DELETE IGNORE FROM `confluence`.`_sbtest3_new` WHERE `confluence`.`_sbtest3_new`.`id` <=> OLD.`id`CREATE TRIGGER `pt_osc_confluence_sbtest3_upd` AFTER UPDATE ON `confluence`.`sbtest3` FOR EACH ROW REPLACE INTO `confluence`.`_sbtest3_new` (`id`, `k`, `c`, `pad`) VALUES (NEW.`id`, NEW.`k`, NEW.`c`, NEW.`pad`)CREATE TRIGGER `pt_osc_confluence_sbtest3_ins` AFTER INSERT ON `confluence`.`sbtest3` FOR EACH ROW REPLACE INTO `confluence`.`_sbtest3_new` (`id`, `k`, `c`, `pad`) VALUES (NEW.`id`, NEW.`k`, NEW.`c`, NEW.`pad`)2016-05-24T20:54:23 Created triggers OK.2016-05-24T20:54:23 Copying approximately 4485573 rows... ==> 分块拷贝数据到新表INSERT LOW_PRIORITY IGNORE INTO `confluence`.`_sbtest3_new` (`id`, `k`, `c`, `pad`) SELECT `id`, `k`, `c`, `pad` FROM `confluence`.`sbtest3` FORCE INDEX(`PRIMARY`) WHERE ((`id` >= ?)) AND ((`id` <= ?)) LOCK IN SHARE MODE /*pt-online-schema-change 44155 copy nibble*/SELECT /*!40001 SQL_NO_CACHE */ `id` FROM `confluence`.`sbtest3` FORCE INDEX(`PRIMARY`) WHERE ((`id` >= ?)) ORDER BY `id` LIMIT ?, 2 /*next chunk boundary*/Copying `confluence`.`sbtest3`: 36% 00:52 remainCopying `confluence`.`sbtest3`: 69% 00:26 remain2016-05-24T20:56:01 Copied rows OK.2016-05-24T20:56:01 Analyzing new table...2016-05-24T20:56:01 Swapping tables... ==> 交换新旧表RENAME TABLE `confluence`.`sbtest3` TO `confluence`.`_sbtest3_old`, `confluence`.`_sbtest3_new` TO `confluence`.`sbtest3`2016-05-24T20:56:01 Swapped original and new tables OK.2016-05-24T20:56:01 Dropping old table... ==> 删除旧表DROP TABLE IF EXISTS `confluence`.`_sbtest3_old`2016-05-24T20:56:02 Dropped old table `confluence`.`_sbtest3_old` OK.2016-05-24T20:56:02 Dropping triggers...DROP TRIGGER IF EXISTS `confluence`.`pt_osc_confluence_sbtest3_del`;DROP TRIGGER IF EXISTS `confluence`.`pt_osc_confluence_sbtest3_upd`;DROP TRIGGER IF EXISTS `confluence`.`pt_osc_confluence_sbtest3_ins`;2016-05-24T20:56:02 Dropped triggers OK.Successfully altered `confluence`.`sbtest3`.
pt-online-schema-change h=10.0.201.34,P=3306,u=jacky,p=xxx,D=confluence,t=sbtest3 \--alter "CHANGE pad f_pad varchar(60) NOT NULL DEFAULT '' " \--print --dry-runpt-online-schema-change -ujacky -p xxx -h "10.0.201.34" D=confluence,t=sbtest3 \--alter "CHANGE pad f_pad varchar(60) NOT NULL DEFAULT '' " \--execute
pt-online-schema-change --user=user --ask-pass --host=10.0.201.34 \--alter "DROP KEY cid, ADD KEY idx_corpid_userid(f_corp_id,f_user_id) " \D=confluence,t=sbtest3 --print --execute
在我的环境里有不少表设计之初没有自增id,而是采用复合主键,pt-osc 对删除、添加主键会特殊处理,详见 。
1. 存在trigger
[zx@mysql-5 ~]$ pt-online-schema-change -u user -p password -h 10.0.200.195 \--alter="MODIFY COLUMN f_receiver varchar(128) NOT NULL DEFAULT '' AFTER f_user_id" --dry-run D=db_name,t=table_nameThe table `db_name`.`table_name` has triggers. This tool needs to create its own triggers, so the table cannot already have triggers.
表上存在触发器,不适用。
2. no-version-check
$ pt-online-schema-change -uuser -ppassword --alter "add key id_provice(f_provice)" \D=db_name,t=tb_name -h rdsxxxxxx.mysql.rds.aliyuncs.comCan't use an undefined value as an ARRAY reference at /usr/bin/pt-online-schema-change line 7335.
这个错误在阿里云RDS上执行时出现的,我以为是我哪里语法写错了,但拿到原生5.6的版本上就没问题了,加上--no-version-check
选项就好了,见 ,没深究,应该是pt去验证mysql server版本的时候从rds拿到的信息不对,导致格式出错。
原文链接地址:
转载地址:http://jybga.baihongyu.com/