MySQL数据库之集合类型SET的DDL变更测试总结
【导读】
针对四种数据类型:布尔类型BOOL或称布尔类型BOOLEAN、微整型TINYTINT、枚举类型ENUM、集合类型SET,我们已经分多篇文章篇幅给出详细的介绍与功能测试数据,接下来我们深入介绍枚举类型EUNM和集合类型SET。测试基于InnoDB存储引擎上,对MySQL数据库集合类型SET的字段进行DDL变更操作,是否需要重新创建表呢?对数据库的事务处理有何影响?对数据库的数据服务提供有何性能影响?
(一) 系统环境
硬件:DELL R510 10块盘做的RAID5,上面跑了几十个虚拟机
操作系统:CentOS release 5.5 (Final)
MySQL数据库:5.5.15-log
InnoDB存储引擎:plugin-InnoDB 1.1.8
(二) 测试数据准备
鉴于测试数据准备需要用到MySQL函数和MySQL存储过程,SQL代码段需要占据文章大量的篇幅,不利于读者阅读关于集合类型SET字段上的DDL变更测试,此部分内容单整理成一篇文章MySQL数据库之数据类型集合类型和枚举类型测试环境。
(三) 集合类型SET字段DDL变更
a) 去掉集合类型字段定义的默认值属性
root@localhost : mysqlops 03:01:38> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','hangzhou','shenzhen','guangzhou','xiamen','tianjin','qingdao','dalian','xian','other') NOT NULL; Query OK, 0 rows affected (0.00 sec) Records: 0 Duplicates: 0 Warnings: 0
小结:
MySQL数据库集合类型字段定义属性为制定了非NULL且有缺省的默认值,对其进行DDL变更,单独去掉缺省的默认值属性,不会出现锁表、重建表等操作,这符合MySQL数据库表DDL变更的特性。
b) 去掉集合类型字段定义的NOT NULL 属性
root@localhost : mysqlops 03:04:48> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','hangzhou','shenzhen','guangzhou','xiamen','tianjin','qingdao','dalian','xian','other'); Query OK, 20017251 rows affected (2 min 4.40 sec) Records: 20017251 Duplicates: 0 Warnings: 0
小结:
MySQL数据库表字段的非NULL属性,修改为默认NULL属性,则会导致锁表与表重建,对于MySQL数据库集合类型字段也同样存在此问题。
c) 增加集合类型字段定义的默认值
root@localhost : mysqlops 03:07:04> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','hangzhou','shenzhen','guangzhou','xiamen','tianjin','qingdao','dalian','xian','other') NOT NULL DEFAULT 'shanghai'; Query OK, 20017251 rows affected (2 min 4.79 sec) Records: 20017251 Duplicates: 0 Warnings: 0
小结:
MySQL数据库表字段属性为NULL,修改为非NULL且有指定的缺省默认值,集合类型字段的DDL变更,也会导致锁表、重新创建等。
d) 修改集合类型字段定义的默认值属性
root@localhost : mysqlops 03:11:44> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','hangzhou','shenzhen','guangzhou','xiamen','tianjin','qingdao','dalian','xian','other') NOT NULL DEFAULT 'beijing'; Query OK, 0 rows affected (0.01 sec) Records: 0 Duplicates: 0 Warnings: 0
小结:
对MySQL数据库表集合类型字段属性为非NULL且有缺省的默认值,仅仅是修改集合类型字段缺省的默认,MySQL数据库不会出现锁表等情况。
e) 修改集合类型字段定义的默认值,且新默认值不在集合列表中
root@localhost : mysqlops 03:12:42> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','hangzhou','shenzhen','guangzhou','xiamen','tianjin','qingdao','dalian','xian','other') NOT NULL DEFAULT 'suzhou'; ERROR 1067 (42000): Invalid default value for 'Work_City'
小结:
修改MySQL数据库集合类型字段缺省的默认值,为集合类型字段值域列表中不存在的集合元素,MySQL数据库回报错,并且集合类型字段的DDL变更SQL语句执行失败。
f) 修改集合类型字段定义,尾部追加集合元素
root@localhost : mysqlops 03:12:52> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','hangzhou','shenzhen','guangzhou','xiamen','tianjin','qingdao','dalian','xian','other','nanchang') NOT NULL DEFAULT 'xian'; Query OK, 0 rows affected (0.00 sec) Records: 0 Duplicates: 0 Warnings: 0
小结:
修改MySQL数据库集合类型字段的值域列表,为值域列表增加集合元素,以值域列表尾部追加的方式,这类DDL变更,不会导致MySQL数据库出现锁表等情况。
g) 修改集合类型字段定义,调整集合元素的顺序
root@localhost : mysqlops 02:57:42> SELECT * FROM mysqlops_set_enum WHERE Work_City=4 LIMIT 1; +----+-------------+-----------+ | ID | Work_Option | Work_City | +----+-------------+-----------+ | 44 | | hangzhou | +----+-------------+-----------+ 1 row in set (0.00 sec) root@localhost : mysqlops 03:13:03> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','xiamen','hangzhou','shenzhen','guangzhou','tianjin','qingdao','dalian','xian','other','nanchang') NOT NULL DEFAULT 'xian'; Query OK, 20017251 rows affected (2 min 20.34 sec) Records: 20017251 Duplicates: 0 Warnings: 0
备注:
DDL变更调整了集合元素 ‘xiamen’ 在集合元素值域列表中的位置,由原来存储序号32,调整为新的存储序号4,而涉及对比的集合元素 ‘hangzhou’ 由存储序号4,调整为新的存储序号8。
root@localhost : mysqlops 03:14:10> SELECT * FROM mysqlops_set_enum WHERE Work_City=4 LIMIT 1; +----+-------------+-----------+ | ID | Work_Option | Work_City | +----+-------------+-----------+ | 12 | | xiamen | +----+-------------+-----------+ 1 row in set (0.00 sec) root@localhost : mysqlops 03:15:05> SELECT * FROM mysqlops_set_enum WHERE ID=44; +----+-------------+-----------+ | ID | Work_Option | Work_City | +----+-------------+-----------+ | 44 | | hangzhou | +----+-------------+-----------+ 1 row in set (0.00 sec) root@localhost : mysqlops 03:16:14> SELECT * FROM mysqlops_set_enum WHERE ID=44 and Work_City=4; Empty set (0.00 sec) root@localhost : mysqlops 03:17:18> SELECT * FROM mysqlops_set_enum WHERE ID=44 and Work_City=8; +----+-------------+-----------+ | ID | Work_Option | Work_City | +----+-------------+-----------+ | 44 | | hangzhou | +----+-------------+-----------+ 1 row in set (0.00 sec) root@localhost : mysqlops 03:18:28> SELECT * FROM mysqlops_set_enum WHERE Work_City=1 LIMIT 1; +----+-------------+-----------+ | ID | Work_Option | Work_City | +----+-------------+-----------+ | 1 | | shanghai | +----+-------------+-----------+ 1 row in set (0.00 sec)
小结:
我们事先随机显示了一条集合元素等于“hangzhou”的记录,自增序列唯一主键值为:44,方便我们对比MySQL数据库集合类型字段DDL变更之后的数据变化,总结如下:
l MySQL数据库集合类型字段的定义属性的集合元素顺序调整,会导致MySQL数据库申请表级锁,并且锁表、创建临时文件等操作;
l 被DDL变更调整顺序的集合元素,其在MySQL数据库集合类型字段属性存储序号发生变化;
l MySQL数据库集合类型字段的定义属性的集合元素顺序调整,导致受其影响的集合元素,字段定义属性中存储顺序都发生变化,数据库表对应的记录值的存储序号也发生变化;
l MySQL数据库集合类型字段的定义属性的某个集合元素顺序调整,对其他存储序号未改变的集合元素,对其对应表字段存储的数据序号编号也未发生变化;
h) 修改集合类型字段定义,删除某个集合元素
删除集合元素”shenzhen”之前的数据库表数据部分参照信息
root@localhost : mysqlops 02:07:28> SELECT COUNT(*) FROM mysqlops_set_enum WHERE Work_city=''; +----------+ | COUNT(*) | +----------+ | 200559 | +----------+ root@localhost : mysqlops 02:20:39> SELECT COUNT(*) FROM mysqlops_set_enum WHERE Work_City='shenzhen'; +----------+ | COUNT(*) | +----------+ | 600160 | +----------+ 1 row in set (7.82 sec) root@localhost : mysqlops 02:25:00> SELECT * FROM mysqlops_set_enum WHERE Work_city='shenzhen' limit 1; +-----+-------------+-----------+ | ID | Work_Option | Work_City | +-----+-------------+-----------+ | 117 | | shenzhen | +-----+-------------+-----------+ 1 row in set (0.00 sec)
执行删除集合元素值“shenzhen”的DDL变更
root@localhost : mysqlops 03:16:05> ALTER TABLE mysqlops_set_enum MODIFY Work_City SET('shanghai','beijing','xiamen','hangzhou','guangzhou','tianjin','qingdao','dalian','xian','other','nanchang') NOT NULL DEFAULT 'xian'; Query OK, 20017251 rows affected, 65535 warnings (2 min 22.09 sec) Records: 20017251 Duplicates: 0 Warnings: 3801649 root@localhost : mysqlops 03:18:48> SHOW WARNINGS; +---------+------+--------------------------------------------------+ | Level | Code | Message | +---------+------+--------------------------------------------------+ | Warning | 1265 | Data truncated for column 'Work_City' at row 11 | | Warning | 1265 | Data truncated for column 'Work_City' at row 19 | | Warning | 1265 | Data truncated for column 'Work_City' at row 34 | | Warning | 1265 | Data truncated for column 'Work_City' at row 41 | | Warning | 1265 | Data truncated for column 'Work_City' at row 47 | | Warning | 1265 | Data truncated for column 'Work_City' at row 50 | | Warning | 1265 | Data truncated for column 'Work_City' at row 62 | | Warning | 1265 | Data truncated for column 'Work_City' at row 69 | | Warning | 1265 | Data truncated for column 'Work_City' at row 76 | | Warning | 1265 | Data truncated for column 'Work_City' at row 82 |
比对删除集合元素”shenzhen”之后的数据库表数据变化
root@localhost : mysqlops 03:19:14> SELECT COUNT(*) FROM mysqlops_set_enum WHERE Work_city=''; +----------+ | COUNT(*) | +----------+ | 800719 | +----------+ 1 row in set (6.29 sec) root@localhost : mysqlops 03:24:09> SELECT * FROM mysqlops_set_enum WHERE ID=117; +-----+-------------+-----------+ | ID | Work_Option | Work_City | +-----+-------------+-----------+ | 117 | | | +-----+-------------+-----------+ 1 row in set (0.00 sec)
小结:
DDL变更SQL语句删除MySQL数据库集合类型字段定义属性的某个集合元素,且该集合元素有对应的数据行存储于数据库表中,将会导致:
l MySQL数据库表需要表级别锁,进行锁住表、重建表等操作;
l 被删除集合元素,对应的数据库表记得行字段的值,将会出现截断,以空字符串替代;
l 被删除集合元素之后的集合元素存储序号发生变化;
l 数据库表中集合类型字段的数据值,为保证对应集合类型字段定义属性中的集合元素存储序号,也会做相应的调整;
(四) 基于集合类型SET字段的索引
a) 完成DDL变更之后的数据库表结构
root@localhost : mysqlops 03:25:42> SHOW CREATE TABLE mysqlops_set_enum\\G *************************** 1. row *************************** Table: mysqlops_set_enum Create Table: CREATE TABLE `mysqlops_set_enum` ( `ID` int(11) NOT NULL AUTO_INCREMENT, `Work_Option` enum('JavaScript','DBA','SA','C++','NA','QA','Java','other','','Python') NOT NULL DEFAULT 'DBA', `Work_City` set('shanghai','beijing','xiamen','hangzhou','guangzhou','tianjin','qingdao','dalian','xian','other','nanchang') NOT NULL DEFAULT 'xian', PRIMARY KEY (`ID`) ) ENGINE=InnoDB AUTO_INCREMENT=20017252 DEFAULT CHARSET=utf8 1 row in set (0.00 sec)
b) MySQL数据库集合类型字段无创建索引时,SQL语句执行计划
root@localhost : mysqlops 03:34:59> EXPLAIN SELECT * FROM mysqlops_set_enum WHERE Work_City=8 LIMIT 1\\G *************************** 1. row *************************** id: 1 select_type: SIMPLE table: mysqlops_set_enum type: ALL possible_keys: NULL key: NULL key_len: NULL ref: NULL rows: 20017710 Extra: Using where 1 row in set (0.00 sec)
小结:
MySQL数据库表集合类型字段没有创建索引时,集合类型字段作为数据查找条件,完全符合MySQL数据库使用索引的规则,依然是全表扫描,说明集合类型字段定义属性的内部存储序列编号,不能起到数据库索引的功效。
c) MySQL数据库集合类型字段创建索引
root@localhost : mysqlops 03:35:44> ALTER TABLE mysqlops_set_enum ADD INDEX idx_work_city_set(Work_City); Query OK, 0 rows affected (1 min 32.40 sec) Records: 0 Duplicates: 0 Warnings: 0
小结:
为MySQL数据库表集合类型字段创建索引,与文章MySQL 5.5版本对普通索引增删性能的优化描述一样,需要锁表、重新创建表等操作,集合类型字段创建索引也同样需要这样做。
d) MySQL数据库集合类型字段有索引时,SQL语句执行计划
root@localhost : mysqlops 03:38:04> EXPLAIN SELECT * FROM mysqlops_set_enum WHERE Work_City=8 LIMIT 1\\G *************************** 1. row *************************** id: 1 select_type: SIMPLE table: mysqlops_set_enum type: ref possible_keys: idx_work_city_set key: idx_work_city_set key_len: 2 ref: const rows: 2024666 Extra: Using where 1 row in set (0.00 sec)
小结:
MySQL数据库表集合类型字段创建数据库索引之后,同样的数据查找SQL语句,则会根据索引条件查找数据,而不是全表扫描。
e) MySQL数据库集合类型字段上的索引删除
root@localhost : mysqlops 03:44:59> ALTER TABLE mysqlops_set_enum DROP INDEX idx_work_city_set; Query OK, 0 rows affected (0.82 sec) Records: 0 Duplicates: 0 Warnings: 0
小结:
MySQL数据库表集合类型字段上的索引删除DDL变更,同样会导致MySQL数据库使用表级锁,锁表、重新创建表等一系列操作。
(五) 总结
对MySQL数据库集合类型的字段定义属性,进行了定义属性默认值去除、NOT NULL 修改为NULL 、默认为NULL修改为NOT NULL和缺省默认值、修改缺省的默认值、尾部追加新的集合元素、调整集合元素的位置、为集合元素字段创建索引、观察数据查找SQL语句执行计划、删除索引等一系列的DDL操作,以及观察数据库表中的集合数据变化,我们得出下列结论:
a) 集合类型字段的DDL变更,对MySQL数据库的影响基本上与其他数据类型字段的DDL变更类似;
b) 集合类型字段的集合元素顺序调整,会导致受影响的集合元素存储编号调整,对应的数据库表中的数据编号也会被修改;
c) 集合类型字段的集合元素顺序调整,不会跟枚举类型字段一样,出现数据库表数据对照关系的紊乱;
d) 集合类型字段的集合元素与表存储的顺序编号之间的对照关系,无法起到索引查找数据的功能。需要时,推荐为数据库表集合类型字段创建索引;
e) 鉴于集合类型字段的集合元素顺序调整,会导致数据库表锁住等操作,建议不要轻易调整集合元素的顺序;
f) 集合类型字段的集合元素增加,最好尾部追加的方式增加,否则需要锁表等一系列操作,影响数据库表的事务处理;
g) 删除集合类型字段定义的某个集合元素,会导致数据库表中对应的数据行发生截断行为,并且用空字符串替代;
h) 集合类型字符定义属性,最好申明为NOT NULL,且有缺省的默认值数据;
数据类型系列讲解文章列表:
MySQL数据库数据类型之ENUM、SET、BOOL/BOOLEAN、TINYINT
MySQL数据库之数据类型BOOL/BOOLEAN与TINYINT测试总结
建议继续学习:
- MySQL数据库之枚举数据类型ENUM的DDL变更测试 (阅读:3820)
- MySQL数据库数据类型之ENUM、SET、BOOL/BOOLEAN、TINYINT (阅读:2877)
- ORACLE 11g新特性-允许DDL锁等待DML锁 (阅读:2428)
- MySQL数据库数据类型之集合类型SET测试总结 (阅读:1496)
扫一扫订阅我的微信号:IT技术博客大学习
- 作者:MySQLOPS 数据库与运维自动化技术分享 来源: MySQLOPS 数据库与运维自动化技术分享
- 标签: DDL SET
- 发布时间:2012-03-19 23:40:40
- [56] WEB系统需要关注的一些点
- [50] Go Reflect 性能
- [50] Oracle MTS模式下 进程地址与会话信
- [48] find命令的一点注意事项
- [47] 图书馆的世界纪录
- [47] Twitter/微博客的学习摘要
- [47] 如何拿下简短的域名
- [46] IOS安全–浅谈关于IOS加固的几种方法
- [45] android 开发入门
- [44] 关于恐惧的自白