在功能開發完畢,在本地或者測試環境進行測試時,常常會遇到這種狀況:有專門的測試數據,測試過程會涉及到修改表中的數據,常常不能一次測試成功,因此,每次執行測試後,原來表中的數據其實已經被修改了,下一次測試,就須要將數據恢復。mysql
我通常的作法是:先建立一個副本表,好比測試使用的user表,我在測試前建立副本表user_bak,每次測試後,將user表清空,而後將副本表user_bak的數據導入到user表中。sql
上面的操做是對一個table作備份,若是涉及到的table太多,能夠建立database的副本。數據庫
接下來我將對此處的表結構複製以及表數據複製進行闡述,並不是數據庫的複製原理!!!!測試
下面是staff表的表結構spa
create table staff ( id int not null auto_increment comment '自增id', name char(20) not null comment '用戶姓名', dep char(20) not null comment '所屬部門', gender tinyint not null default 1 comment '性別:1男; 2女', addr char(30) not null comment '地址', primary key(id), index idx_1 (name, dep), index idx_2 (name, gender) ) engine=innodb default charset=utf8mb4 comment '員工表';
若是原始表已經存在,那麼能夠使用命令查看該表的建立語句:blog
mysql> show create table staff\G *************************** 1. row *************************** Table: staff Create Table: CREATE TABLE `staff` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增id', `name` char(20) NOT NULL COMMENT '用戶姓名', `dep` char(20) NOT NULL COMMENT '所屬部門', `gender` tinyint(4) NOT NULL DEFAULT '1' COMMENT '性別:1男; 2女', `addr` char(30) NOT NULL, PRIMARY KEY (`id`), KEY `idx_1` (`name`,`dep`), KEY `idx_2` (`name`,`gender`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='員工表' 1 row in set (0.01 sec)
能夠看到,上面show creat table xx的命令執行結果中,Create Table的值就是建立表的語句,此時能夠直接複製建立表的SQL,而後從新執行一次就好了。索引
當數據表中有數據的時候,看到的建立staff表的sql就會稍有不一樣。好比,我在staff中添加了兩條記錄:開發
mysql> insert into staff values (null, '李明', 'RD', 1, '北京'); Query OK, 1 row affected (0.00 sec) mysql> insert into staff values (null, '張三', 'PM', 0, '上海'); Query OK, 1 row affected (0.00 sec) mysql> select * from staff; +----+--------+-----+--------+--------+ | id | name | dep | gender | addr | +----+--------+-----+--------+--------+ | 1 | 李明 | RD | 1 | 北京 | | 2 | 張三 | PM | 0 | 上海 | +----+--------+-----+--------+--------+ 2 rows in set (0.00 sec)
此時在執行show create table命令:rem
mysql> show create table staff\G *************************** 1. row *************************** Table: staff Create Table: CREATE TABLE `staff` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增id', `name` char(20) NOT NULL COMMENT '用戶姓名', `dep` char(20) NOT NULL COMMENT '所屬部門', `gender` tinyint(4) NOT NULL DEFAULT '1' COMMENT '性別:1男; 2女', `addr` char(30) NOT NULL, PRIMARY KEY (`id`), KEY `idx_1` (`name`,`dep`), KEY `idx_2` (`name`,`gender`) ) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COMMENT='員工表' 1 row in set (0.00 sec)
注意,上面結果中的倒數第二行innodb
ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COMMENT='員工表'
由於staff表的id是自增的,且已經有了2條記錄,因此下一次插入數據的自增id應該爲3,這個信息,也會出如今表的建立sql中。
使用like根據已有的表來建立新表,特色以下:
一、方便,不須要查看原表的表結構定義信息;
二、建立的新表中,表結構定義、完整性約束,都與原表保持一致。
三、建立的新表是一個空表,全新的表,沒有數據。
用法以下:
mysql> select * from staff; #舊錶中已有2條數據 +----+--------+-----+--------+--------+ | id | name | dep | gender | addr | +----+--------+-----+--------+--------+ | 1 | 李明 | RD | 1 | 北京 | | 2 | 張三 | PM | 0 | 上海 | +----+--------+-----+--------+--------+ 2 rows in set (0.00 sec) mysql> create table staff_bak_1 like staff; # 直接使用like,前面指定新表名,後面指定舊錶(參考的表) Query OK, 0 rows affected (0.02 sec) mysql> show create table staff_bak_1\G *************************** 1. row *************************** Table: staff_bak_1 Create Table: CREATE TABLE `staff_bak_1` ( `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增id', `name` char(20) NOT NULL COMMENT '用戶姓名', `dep` char(20) NOT NULL COMMENT '所屬部門', `gender` tinyint(4) NOT NULL DEFAULT '1' COMMENT '性別:1男; 2女', `addr` char(30) NOT NULL, PRIMARY KEY (`id`), KEY `idx_1` (`name`,`dep`), KEY `idx_2` (`name`,`gender`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='員工表' # 注意沒有AUTO_INCREMENT=3 1 row in set (0.00 sec) mysql> select * from staff_bak_1; # 沒有包含舊錶的數據 Empty set (0.00 sec)
使用as來建立新表,有一下特色:
一、能夠有選擇性的決定新表包含哪些字段;
二、建立的新表中,會包含舊錶的數據;
三、建立的新表不會包含舊錶的完整性約束(好比主鍵、索引等),僅包含最基礎的表結構定義。
用法以下:
mysql> create table staff_bak_2 as select * from staff; Query OK, 2 rows affected (0.02 sec) Records: 2 Duplicates: 0 Warnings: 0 mysql> select * from staff_bak_2; +----+--------+-----+--------+--------+ | id | name | dep | gender | addr | +----+--------+-----+--------+--------+ | 1 | 李明 | RD | 1 | 北京 | | 2 | 張三 | PM | 0 | 上海 | +----+--------+-----+--------+--------+ 2 rows in set (0.00 sec) mysql> show create table staff_bak_2\G *************************** 1. row *************************** Table: staff_bak_2 Create Table: CREATE TABLE `staff_bak_2` ( `id` int(11) NOT NULL DEFAULT '0' COMMENT '自增id', `name` char(20) CHARACTER SET utf8mb4 NOT NULL COMMENT '用戶姓名', `dep` char(20) CHARACTER SET utf8mb4 NOT NULL COMMENT '所屬部門', `gender` tinyint(4) NOT NULL DEFAULT '1' COMMENT '性別:1男; 2女', `addr` char(30) CHARACTER SET utf8mb4 NOT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 1 row in set (0.00 sec)
利用as建立表的時候沒有保留完整性約束,其實這個仔細想一下也能想明白。由於使用as建立表的時候,能夠指定新表包含哪些字段呀,若是你建立新表時,忽略了幾個字段,這樣的話即便保留了完整約束,保存數據是也不能知足完整性約束。
好比,staff表有一個索引idx1,由name和dep字段組成;可是我建立的新表中,沒有name和dep字段(只選擇了其餘字段),那麼新表中保存idx1也沒有必要,對吧。
mysql> -- 只選擇id、gender、addr做爲新表的字段,那麼name和dep組成的索引就不必存在了 mysql> create table staff_bak_3 as (select id, gender, addr from staff); Query OK, 2 rows affected (0.02 sec) Records: 2 Duplicates: 0 Warnings: 0 mysql> show create table staff_bak_3\G *************************** 1. row *************************** Table: staff_bak_3 Create Table: CREATE TABLE `staff_bak_3` ( `id` int(11) NOT NULL DEFAULT '0' COMMENT '自增id', `gender` tinyint(4) NOT NULL DEFAULT '1' COMMENT '性別:1男; 2女', `addr` char(30) CHARACTER SET utf8mb4 NOT NULL ) ENGINE=InnoDB DEFAULT CHARSET=latin1 1 row in set (0.00 sec) mysql> select * from staff_bak_3; +----+--------+--------+ | id | gender | addr | +----+--------+--------+ | 1 | 1 | 北京 | | 2 | 0 | 上海 | +----+--------+--------+ 2 rows in set (0.00 sec)
使用like建立新表,雖然保留了舊錶的各類表結構定義以及完整性約束,可是如何將舊錶的數據導入到新表中呢?
最極端的方式:寫一個程序,先將舊錶數據讀出來,而後寫入到新表中,這個方式我就不嘗試了。
有一個比較簡單的命令:
mysql> select * from staff; #原表數據 +----+--------+-----+--------+--------+ | id | name | dep | gender | addr | +----+--------+-----+--------+--------+ | 1 | 李明 | RD | 1 | 北京 | | 2 | 張三 | PM | 0 | 上海 | +----+--------+-----+--------+--------+ 2 rows in set (0.00 sec) mysql> select * from staff_bak_1; # 使用like建立的表,與原表相同的表結構和完整性約束(自增除外) Empty set (0.00 sec) mysql> insert into staff_bak_1 select * from staff; # 將staff表的全部記錄的全部字段值都插入副本表中 Query OK, 2 rows affected (0.00 sec) Records: 2 Duplicates: 0 Warnings: 0 mysql> select * from staff_bak_1; +----+--------+-----+--------+--------+ | id | name | dep | gender | addr | +----+--------+-----+--------+--------+ | 1 | 李明 | RD | 1 | 北京 | | 2 | 張三 | PM | 0 | 上海 | +----+--------+-----+--------+--------+ 2 rows in set (0.00 sec)
其實這條SQL語句,是知道兩個表的表結構和完整性約束相同,因此,能夠直接select *。
insert into staff_bak_1 select * from staff;
若是兩個表結構不相同,其實也是能夠這個方式的,好比:
mysql> show create table demo\G *************************** 1. row *************************** Table: demo Create Table: CREATE TABLE `demo` ( `_id` int(11) NOT NULL AUTO_INCREMENT, `_name` char(20) DEFAULT NULL, `_gender` tinyint(4) DEFAULT '1', PRIMARY KEY (`_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 1 row in set (0.00 sec) # 只將staff表中的id和name字段組成的數據記錄插入到demo表中,對應_id和_name字段 mysql> insert into demo (_id, _name) select id,name from staff; Query OK, 2 rows affected (0.00 sec) Records: 2 Duplicates: 0 Warnings: 0 mysql> select * from demo; +-----+--------+---------+ | _id | _name | _gender | +-----+--------+---------+ | 1 | 李明 | 1 | | 2 | 張三 | 1 | +-----+--------+---------+ 2 rows in set (0.00 sec)
這是兩個表的字段數量不相同的狀況,此時須要手動指定列名,不然就會報錯。
另外,若是兩個表的字段數量,以及相同順序的字段類型相同,若是是所有字段複製,即便字段名不一樣,也能夠直接複製:
# staff_bak_5的字段名與staff表並不相同,可是字段數量、相同順序字段的類型相同,因此能夠直接插入 mysql> show create table staff_bak_5\G *************************** 1. row *************************** Table: staff_bak_5 Create Table: CREATE TABLE `staff_bak_5` ( `_id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增id', `_name` char(20) NOT NULL COMMENT '用戶姓名', `_dep` char(20) NOT NULL COMMENT '所屬部門', `_gender` tinyint(4) NOT NULL DEFAULT '1' COMMENT '性別:1男; 2女', `_addr` char(30) NOT NULL, PRIMARY KEY (`_id`), KEY `idx_1` (`_name`,`_dep`), KEY `idx_2` (`_name`,`_gender`) ) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COMMENT='員工表' 1 row in set (0.00 sec) mysql> insert into staff_bak_5 select * from staff; Query OK, 2 rows affected (0.00 sec) Records: 2 Duplicates: 0 Warnings: 0 mysql> select * from staff_bak_5; +-----+--------+------+---------+--------+ | _id | _name | _dep | _gender | _addr | +-----+--------+------+---------+--------+ | 1 | 李明 | RD | 1 | 北京 | | 2 | 張三 | PM | 0 | 上海 | +-----+--------+------+---------+--------+ 2 rows in set (0.00 sec)