av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

您的位置:首頁技術(shù)文章
文章詳情頁

MySQL 大表添加一列的實(shí)現(xiàn)

瀏覽:7日期:2023-10-06 08:06:22

問題參考自: https://www.zhihu.com/question/440231149 ,mysql中,一張表里有3億數(shù)據(jù),未分表,要求是在這個(gè)大表里添加一列數(shù)據(jù)。數(shù)據(jù)庫不能停,并且還有增刪改操作。請問如何操作?答案為個(gè)人原創(chuàng)

以前老版本 MySQL 添加一列的方式:

ALTER TABLE 你的表 ADD COLUMN 新列 char(128);

會造成鎖表,簡易過程如下:

新建一個(gè)和 Table1 完全同構(gòu)的 Table2 對表 Table1 加寫鎖 在表 Table2 上執(zhí)行 ALTER TABLE 你的表 ADD COLUMN 新列 char(128) 將 Table1 中的數(shù)據(jù)拷貝到 Table2 將 Table2 重命名為 Table1 并移除 Table1,釋放所有相關(guān)的鎖

如果數(shù)據(jù)量特別特別大,那么鎖表時(shí)間很長,期間所有表更新都會阻塞,線上業(yè)務(wù)不能正常執(zhí)行。

針對 MySQL 5.6(不包含)之前的版本,通過觸發(fā)器將一個(gè)表的更新在另一個(gè)表上重復(fù),并進(jìn)行數(shù)據(jù)同步,當(dāng)數(shù)據(jù)同步完成時(shí),業(yè)務(wù)上修改表名為新表并發(fā)布。業(yè)務(wù)不會暫停。觸發(fā)器設(shè)置類似于:

create trigger person_trigger_update AFTER UPDATE on 原有表 for each row begin set @x = 'trigger UPDATE';Replace into 新表 SELECT * from 原有表 where 新表.id = 原有表.id;END IF;end;

MySQL 5.6(包含) 以后的版本引入了在線 DDL 的功能:

Alter table 你的表 , ALGORITHM [=] {DEFAULT|INSTANT|INPLACE|COPY}, LOCK [=] { DEFAULT| NONE| SHARED| EXCLUSIVE }

其中的參數(shù):

ALGORITHM:

DEFAULT:默認(rèn)方式,在 MySQL 8.0中,如果未顯示指定 ALGORITHM,那么會優(yōu)先選擇 INSTANT 算法,如果不行再使用 INPLACE 算法,如果不支持 INPLACE 算法則使用 COPY 的方式完成 INSTANT:8.0 中新添加的算法,添加列是立即返回。但是不能是虛擬列。這個(gè)原理很簡單,對于新建一列,表所有原有數(shù)據(jù)并不是立刻發(fā)生變化,只是在表字典里面記錄下這個(gè)列和默認(rèn)值,對于默認(rèn)的 Dynamic 行格式(其實(shí)就是 Compressed 的變種),如果更新了這一列則原有數(shù)據(jù)標(biāo)記為刪除在末尾追加更新后的記錄。這樣做就是沒有提前預(yù)留出列空間,之后更新可能經(jīng)常會發(fā)生行記錄空間變動。但是對于大多數(shù)業(yè)務(wù),都是最近的時(shí)間的記錄才會修改,所以問題不大。 INPLACE:在原表上直接進(jìn)行修改,不會拷貝臨時(shí)表,可以逐條記錄修改,不會產(chǎn)生大量的 undolog 以及 redolog,不會占用很多 buffer。可以避免重建表帶來的IO和CPU消耗,保證期間依然良好的性能和并發(fā)。 COPY:拷貝到臨時(shí)新表上進(jìn)行修改。由于記錄拷貝,會產(chǎn)生大量的 undolog 以及 redolog,并占用很多 buffer,對業(yè)務(wù)性能有影響。

LOCK:

DEFAULT:和 ALGORITHM 的 DEFAULT 類似 NONE:無鎖,允許并發(fā)讀取和更新表 SHARED:共享鎖,允許讀取不允許更新 EXCLUSIVE:不允許讀取和更新

各個(gè)版本支持的在線 DDL 修改使用的算法的對比:

MySQL 大表添加一列的實(shí)現(xiàn)

參考文檔:

MySQL 5.6:https://dev.mysql.com/doc/refman/5.6/en/innodb-online-ddl-operations.htmlMySQL

5.7:https://dev.mysql.com/doc/refman/5.7/en/innodb-online-ddl-operations.htmlMySQL

8.0:https://dev.mysql.com/doc/refman/8.0/en/innodb-online-ddl-operations.html

可以通過:

ALTER TABLE 你的表 ADD COLUMN 新列 char(128), ALGORITHM=INSTANT, LOCK=NONE;

類似的語句,實(shí)現(xiàn)在線增加字段。最好還是明確 ALGORITHM 以及 LOCK,這樣執(zhí)行 DDL 的時(shí)候能明確知道到底會對線上業(yè)務(wù)有多大影響

同時(shí),執(zhí)行在線 DDL 的過程大概是:

MySQL 大表添加一列的實(shí)現(xiàn)

可以看出,在開始階段需要 metadata lock,metadata lock 是在 5.5 才引入到mysql,之前也有類似保護(hù)元數(shù)據(jù)的機(jī)制,只是沒有明確提出 metadata lock 概念而已。但是 5.5 之前版本(比如5.1)與5.5之后版本在保護(hù)元數(shù)據(jù)這塊有一個(gè)顯著的不同點(diǎn)是,5.1對于元數(shù)據(jù)的保護(hù)是語句級別的,5.5對于metadata的保護(hù)是事務(wù)級別的。所謂語句級別,即語句執(zhí)行完成后,無論事務(wù)是否提交或回滾,其表結(jié)構(gòu)可以被其他會話更新;而事務(wù)級別則是在事務(wù)結(jié)束后才釋放 metadata lock。

引入 metadata lock 后,主要解決了2個(gè)問題,一個(gè)是事務(wù)隔離問題,比如在可重復(fù)隔離級別下,會話A在2次查詢期間,會話B對表結(jié)構(gòu)做了修改,兩次查詢結(jié)果就會不一致,無法滿足可重復(fù)讀的要求;另外一個(gè)是數(shù)據(jù)復(fù)制的問題,比如會話A執(zhí)行了多條更新語句期間,另外一個(gè)會話B做了表結(jié)構(gòu)變更并且先提交,就會導(dǎo)致 slave 在重做時(shí),先重做 alter,再重做 update 時(shí)就會出現(xiàn)復(fù)制錯(cuò)誤的現(xiàn)象。

如果當(dāng)前有很多事務(wù)在執(zhí)行,并且有那種包含大查詢的事務(wù),例如:

START TRANSACTION;select count(*) from 你的表

這樣類似的會執(zhí)行較長時(shí)間的事務(wù),也會阻塞。

所以,原則上:

避免大事務(wù) 在業(yè)務(wù)低峰去做表結(jié)構(gòu)變化

到此這篇關(guān)于MySQL 大表添加一列的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)MySQL 大表添加一列內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: MySQL 數(shù)據(jù)庫
相關(guān)文章:
主站蜘蛛池模板: 93久久精品日日躁夜夜躁欧美 | 欧美 日韩 国产 成人 在线 | 亚洲激情综合 | 黄色av免费在线观看 | 亚洲一区在线看 | 久久成人精品视频 | 日韩精品视频在线播放 | 亚洲色网址 | 亚洲三级免费 | 中文字幕97| 精品久久久久久久久久久久久 | 成人三级在线观看 | 黄色成人在线 | 毛片毛片毛片 | 日本久久精品视频 | 国产做受网站 | 国内精品久久久久久久久 | 欧美色综合网 | 999毛片| 国产精品久久久久久久久久久久午夜片 | 国产成人av网站 | 日韩国产在线播放 | 99久久精品国产一区二区成人 | 天堂8中文 | 99伊人网 | 久久国产精品免费视频 | 伊人久久免费视频 | 一道本在线视频 | 国产一区二区免费看 | 日日操夜夜干 | 手机av免费 | 一区二区三区成人 | 成人免费网站黄 | www.爱爱| 韩国精品一区 | 夜夜操影院 | 欧美性生活网站 | 精品一区二区三 | 99国产精品99久久久久久 | 欧美不卡 | 日韩一级片 |