MYSQL大表改字段慢問(wèn)題的解決
Mysql如何加快大表的ALTER TABLE操作速度
MYSQL的ALTER TABLE操作的性能對(duì)大表來(lái)說(shuō)是個(gè)大問(wèn)題。MYSQL執(zhí)行大部分修改表結(jié)構(gòu)操作的方法是用新的表結(jié)構(gòu)創(chuàng)建一個(gè)空表,從舊表中查出所有數(shù)據(jù)插入新表,然后刪除舊表。這樣操作可能需要花費(fèi)很長(zhǎng)時(shí)間,如果內(nèi)存不足而表又很大,而且還有很多索引的情況下尤其如此。許多人都有這樣的經(jīng)驗(yàn),ALTER TABLE操作需要花費(fèi)數(shù)個(gè)小時(shí)甚至數(shù)天才能完成。
一般而言,大部分ALTER TABLE操作將導(dǎo)致MYSQL服務(wù)中斷。對(duì)常見(jiàn)的場(chǎng)景,能使用的技巧只有兩種:
- 一種是先在一臺(tái)不提供服務(wù)的機(jī)器上執(zhí)行ALTER TABLE操作,然后和提供服務(wù)的主庫(kù)進(jìn)行切換;
- 另外一種技巧就是“影子拷貝”。影子拷貝技巧是用要求的表結(jié)構(gòu)創(chuàng)建一張新表,然后通過(guò)重命名和刪表操作交換兩張表。
不是所有的ALTER TABLE操作都會(huì)引起表重建。例如,有兩種方法可以改變或刪除一個(gè)列的默認(rèn)值(一種方法很快,另一種則很慢)。
假如要修改電影的默認(rèn)租賃期限,從三天改到五天。下面是很慢的方式:
mysql> ALTER TABLE film modify column rental_duration tinyint(3) not null default 5;
SHOW STATUS顯示這個(gè)語(yǔ)句做了1000次讀和1000次插入操作。換句話說(shuō),它拷貝了整張表到一張新表,甚至列的類型、大小和可否為null屬性都沒(méi)有改變。
理論上,MYSQL可以跳過(guò)創(chuàng)建新表的吧步驟。列的默認(rèn)值實(shí)際上存在表的.frm文件中,所以可以直接修改這個(gè)文件而不需要改動(dòng)表本身。然而MYSQL還沒(méi)有采用這種優(yōu)化的方法,所以MODIFY COLUMN操作都將導(dǎo)致表重建。
另外一種方法是通過(guò)ALTER COLUMN操作來(lái)改變列的默認(rèn)值;
mysql> ALTER TABLE film ALTER COLUMN rental_duration set DEFAULT 5;
這個(gè)語(yǔ)句會(huì)直接修改.frm文件而不涉及表數(shù)據(jù)。所以這個(gè)操作是非??斓?。
只修改.frm文件
從上面的例子我們看到修改表的.frm文件是很快的,但MYSQL有時(shí)候會(huì)在沒(méi)有必要的時(shí)候也重建表。如果愿意冒一些風(fēng)險(xiǎn),可以讓MYSQL做一些其他類型的修改而不用重建表。
注意 下面要演示的技巧是不受官方支持的,也沒(méi)有文檔記錄,并且也可能不能正常工作,采用這些技術(shù)需要自己承擔(dān)風(fēng)險(xiǎn)。>建議在執(zhí)行之前首先備份數(shù)據(jù)!
下面這些操作是有可能不需要重建表的:
- 移除(不是增加)一個(gè)列的AUTO_INCREMENT屬性。
- 增加、移除,或更改ENUM和SET常亮。如果移除的是已經(jīng)有行數(shù)據(jù)用到其值的常量,查詢將會(huì)返回一個(gè)空字符串。
步驟:
- 創(chuàng)建一張有相同結(jié)構(gòu)的空表,并進(jìn)行所需要的修改(例如:增加ENUM常量)。
- 執(zhí)行FLUSH TABLES WITH READ LOCK。這將會(huì)關(guān)閉所有正在使用的表,并且禁止任何表被打開(kāi)。
- 交換.frm文件。
- 執(zhí)行UNLOCK TABLES 來(lái)釋放第二步的讀鎖。
下面以給film表的rating列增加一個(gè)常量為例來(lái)說(shuō)明。當(dāng)前列看起來(lái)如下:
mysql> SHOW COLUMNS FROM film LIKE 'rating';
| Field | Type | Null | Key | Default | Extra |
|---|---|---|---|---|---|
| rating | enum('G','PG','PG-13','R','NC-17') | YES | G |
假設(shè)我們需要為那些對(duì)電影更加謹(jǐn)慎的父母?jìng)冊(cè)黾右粋€(gè)PG-14的電影分級(jí):
mysql> CREATE TABLE film_new like film;
mysql> ALTER TABLE film_new modify column rating ENUM('G','PG','PG-13','R','NC-17','PG-14') DEFAULT 'G';
mysql> FLUSH TABLES WITH READ LOCK;
注意,我們是在常量列表的末尾增加一個(gè)新的值。如果把新增的值放在中間,例如:PG-13之后,則會(huì)導(dǎo)致已經(jīng)存在的數(shù)據(jù)的含義被改變:已經(jīng)存在的R值將變成PG-14,而已經(jīng)存在的NC-17將成為R,等等。
接下來(lái)用操作系統(tǒng)的命令交換.frm文件:
/var/lib/mysql/sakila# mv film.frm film_tmp.frm /var/lib/mysql/sakila# mv film_new.frm film.frm /var/lib/mysql/sakila# mv film_tmp.frm film_new.frm
再回到Mysql命令行,現(xiàn)在可以解鎖表并且看到變更后的效果了:
mysql> UNLOCK TABLES; mysql> SHOW COLUMNS FROM film like 'rating'\G
****************** 1. row*********************
Field: rating
Type: enum('G','PG','PG-13','R','NC-17','PG-14')
最后需要做的是刪除為完成這個(gè)操作而創(chuàng)建的輔助表:
mysql> DROP TABLE film_new;
到此這篇關(guān)于MYSQL大表改字段慢問(wèn)題的解決的文章就介紹到這了,更多相關(guān)MYSQL大表改字段慢內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決出現(xiàn)secure_file_priv null的問(wèn)題
這篇文章主要介紹了解決出現(xiàn)secure_file_priv null的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-03-03
gearman + mysql方式實(shí)現(xiàn)持久化操作示例
這篇文章主要介紹了gearman + mysql方式實(shí)現(xiàn)持久化操作,簡(jiǎn)單描述了持久化的概念、原理,并結(jié)合實(shí)例形式分析了gearman + mysql持久化操作相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2020-02-02
MySQL高效模糊搜索之內(nèi)置函數(shù)locate instr position find_in_set使用詳解
在MySQL中一般進(jìn)行模糊搜索都是使用LIKE配合通配符進(jìn)行查詢的,在性能上一定的影響,下面給大家分享MYSQL自帶的內(nèi)置模糊搜索函數(shù),除最后一個(gè)外其它三個(gè)性能上要比Like快些2018-09-09
如何使用mysql語(yǔ)句進(jìn)行多表聯(lián)查(以三個(gè)表為例)
這篇文章主要介紹了如何使用mysql語(yǔ)句進(jìn)行多表聯(lián)查(以三個(gè)表為例),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
Windows下MySQL 5.6安裝及配置詳細(xì)圖解(大圖版)
這篇文章主要介紹了Windows下MySQL 5.6安裝及配置詳細(xì)圖解(大圖版),需要的朋友可以參考下2016-04-04

