InnoDB存储引擎下快速创建索引

    在MySQL5.0版本前,新增或删除索引会很缓慢,特别是比较大的表。它需要创建一个新的表,然后将旧表中的数据插入到新的表中,更新索引,待这一切完成后再将旧表删掉并把新表的表名改成原来的表名。
    从MySQL5.1版本开始,MySQL允许创建或删除索引不必拷贝整个表内容。在MySQL内置版本的InnoDB存储引擎并没有这种性能;InnoDB Plugin版本可以实现。
    在InnoDB中,表中行存储在聚集索引或主键索引,称之为索引组织表。不管是内置版本的InnoDB存储引擎还是InnoDB Plugin存储引擎,更改聚集索引或主键都需要复制数据。但是添加或删除InnoDB Plugin辅助索引的速度要快得多,因为它不涉及复制数据。
Built-in InnoDB版本下:

 
  1. mysql> CREATE TABLE T1(A INT PRIMARY KEY, B INT, C CHAR(1)) ENGINE=InnoDB; 
  2. mysql> set profiling = 1; 
  3. mysql> ALTER TABLE T1 ADD INDEX (B), ADD UNIQUE INDEX (C); 
  4. mysql> show profile cpu for query 8;          
  5. +----------------------+----------+----------+------------+ 
  6. | Status               | Duration | CPU_user | CPU_system | 
  7. +----------------------+----------+----------+------------+ 
  8. | starting             | 0.000044 | 0.000000 |   0.000000 |  
  9. | checking permissions | 0.000002 | 0.000000 |   0.000000 |  
  10. | checking permissions | 0.000003 | 0.000000 |   0.000000 |  
  11. | init                 | 0.000035 | 0.000000 |   0.000000 |  
  12. | Opening tables       | 0.000056 | 0.000000 |   0.000000 |  
  13. | System lock          | 0.000003 | 0.000000 |   0.000000 |  
  14. Table lock           | 0.000003 | 0.000000 |   0.000000 |  
  15. | setup                | 0.000018 | 0.000000 |   0.000000 |  
  16. | creating table       | 0.003217 | 0.000999 |   0.000000 |  
  17. After create         | 0.000047 | 0.000000 |   0.000000 |  
  18. copy to tmp table    | 0.000034 | 0.000000 |   0.000000 |  
  19. | rename result table  | 0.001110 | 0.000000 |   0.000000 |  
  20. end                  | 0.000038 | 0.000000 |   0.000000 |  
  21. | query end            | 0.000004 | 0.000000 |   0.000000 |  
  22. | freeing items        | 0.000127 | 0.000000 |   0.000000 |  
  23. | cleaning up          | 0.000004 | 0.000000 |   0.000000 |  
  24. +----------------------+----------+----------+------------+ 

删除索引:

 
  1. mysql> alter table T1 drop PRIMARY KEY
  2. mysql> show profile cpu for query 11; 
  3. +----------------------+----------+----------+------------+ 
  4. | Status               | Duration | CPU_user | CPU_system | 
  5. +----------------------+----------+----------+------------+ 
  6. | starting             | 0.000038 | 0.000000 |   0.000000 |  
  7. | checking permissions | 0.000002 | 0.000000 |   0.000000 |  
  8. | checking permissions | 0.000003 | 0.000000 |   0.000000 |  
  9. | init                 | 0.000034 | 0.000000 |   0.000000 |  
  10. | Opening tables       | 0.000010 | 0.000000 |   0.000000 |  
  11. | System lock          | 0.000003 | 0.000000 |   0.000000 |  
  12. Table lock           | 0.000004 | 0.000000 |   0.000000 |  
  13. | setup                | 0.000015 | 0.000000 |   0.000000 |  
  14. | creating table       | 0.003219 | 0.000000 |   0.001000 |  
  15. After create         | 0.000039 | 0.000000 |   0.000000 |  
  16. copy to tmp table    | 0.000029 | 0.000000 |   0.000000 |  
  17. | rename result table  | 0.001346 | 0.001000 |   0.000000 |  
  18. end                  | 0.000038 | 0.000000 |   0.000000 |  
  19. | query end            | 0.000003 | 0.000000 |   0.000000 |  
  20. | freeing items        | 0.000134 | 0.000000 |   0.000000 |  
  21. | cleaning up          | 0.000003 | 0.000000 |   0.000000 |  
  22. +----------------------+----------+----------+------------+ 

InnoDB Plugin版本下:

 
  1. mysql> show variables like "innodb_version"
  2. +----------------+-------+ 
  3. | Variable_name  | Value | 
  4. +----------------+-------+ 
  5. | innodb_version | 1.1.4 | 
  6. +----------------+-------+ 
  7.  
  8. mysql> CREATE TABLE T1(A INT PRIMARY KEY, B INT, C CHAR(1)) ENGINE=InnoDB; 
  9. mysql> set profiling = 1; 
  10. mysql> ALTER TABLE T1 ADD INDEX (B), ADD UNIQUE INDEX (C); 
  11. mysql> show profile cpu for query 16; 
  12. +------------------------------+----------+----------+------------+ 
  13. | Status                       | Duration | CPU_user | CPU_system | 
  14. +------------------------------+----------+----------+------------+ 
  15. | starting                     | 0.000048 | 0.000000 |   0.000000 | 
  16. | checking permissions         | 0.000004 | 0.000000 |   0.000000 | 
  17. | checking permissions         | 0.000003 | 0.000000 |   0.000000 | 
  18. | init                         | 0.000007 | 0.000000 |   0.000000 | 
  19. | Opening tables               | 0.000095 | 0.000000 |   0.000000 | 
  20. | System lock                  | 0.000006 | 0.000000 |   0.000000 | 
  21. | setup                        | 0.000021 | 0.000000 |   0.000000 | 
  22. | creating table               | 0.001577 | 0.000000 |   0.000000 | 
  23. After create                 | 0.000026 | 0.000000 |   0.000000 | 
  24. | manage keys                  | 0.084208 | 0.003000 |   0.010998 | 
  25. | rename result table          | 0.000158 | 0.000000 |   0.000000 | 
  26. end                          | 0.000017 | 0.000000 |   0.000000 | 
  27. | Waiting for query cache lock | 0.000002 | 0.000000 |   0.000000 | 
  28. end                          | 0.000005 | 0.000000 |   0.000000 | 
  29. | query end                    | 0.000003 | 0.000000 |   0.000000 | 
  30. | closing tables               | 0.000007 | 0.000000 |   0.000000 | 
  31. | freeing items                | 0.000013 | 0.000000 |   0.000000 | 
  32. | cleaning up                  | 0.000002 | 0.000000 |   0.000000 | 
  33. +------------------------------+----------+----------+------------+ 

重构聚集索引或主键索引时,InnoDB需要拷贝数据。如下:

 
  1. mysql> CREATE TABLE T2 (A INT, B INT) ENGINE=InnoDB; 
  2. mysql> set profiling = 1; 
  3. mysql> ALTER TABLE T2 ADD PRIMARY KEY (B); 
  4. mysql> show profile cpu for query 1; 
  5. +------------------------------+----------+----------+------------+ 
  6. | Status                       | Duration | CPU_user | CPU_system | 
  7. +------------------------------+----------+----------+------------+ 
  8. | starting                     | 0.000046 | 0.000000 |   0.000000 | 
  9. | checking permissions         | 0.000004 | 0.000000 |   0.000000 | 
  10. | checking permissions         | 0.000004 | 0.000000 |   0.000000 | 
  11. | init                         | 0.000006 | 0.000000 |   0.000000 | 
  12. | Opening tables               | 0.000109 | 0.001000 |   0.000000 | 
  13. | System lock                  | 0.000006 | 0.000000 |   0.000000 | 
  14. | setup                        | 0.000016 | 0.000000 |   0.000000 | 
  15. | creating table               | 0.016539 | 0.000000 |   0.001000 | 
  16. After create                 | 0.000059 | 0.000000 |   0.000000 | 
  17. copy to tmp table            | 0.000040 | 0.000000 |   0.000000 | 
  18. | rename result table          | 0.036043 | 0.000999 |   0.001000 | 
  19. end                          | 0.000042 | 0.000000 |   0.000000 | 
  20. | Waiting for query cache lock | 0.000002 | 0.000000 |   0.000000 | 
  21. end                          | 0.000008 | 0.000000 |   0.000000 | 
  22. | query end                    | 0.000003 | 0.000000 |   0.000000 | 
  23. | closing tables               | 0.000008 | 0.000000 |   0.000000 | 
  24. | freeing items                | 0.000022 | 0.000000 |   0.000000 | 
  25. | cleaning up                  | 0.000003 | 0.000000 |   0.000000 | 
  26. +------------------------------+----------+----------+------------+ 

   在创建UNIQUE或PRIMARY KEY索引时,Innodb还需要做一些额外的工作。对于UNIQUE索引,InnoDB需要检查表不包含重复的键值。对于PRIMARY KEY索引,InnoDB需要检查主键列有没包含NULL。因此最好在创建表的同时,创建主键。

   注意:如何添加或删除REFERENCES外键约束,子表还是需要复制数据的。