問題參考自:https://www.zhihu.com/question/440066129/answer/1685329456 ,mysql中,一張表里有3億數(shù)據(jù),未分表,其中一個字段是企業(yè)類型,企業(yè)類型是一般企業(yè)和個體戶,個體戶的數(shù)據(jù)量差不多占50%,根據(jù)條件把個體戶的行都刪掉。請問如何操作?答案為個人原創(chuàng)
假設(shè)表的引擎是 Innodb, MySQL 5.7+
刪除一條記錄,首先鎖住這條記錄,數(shù)據(jù)原有的被廢棄,記錄頭發(fā)生變化,主要是打上了刪除標(biāo)記。也就是原有的數(shù)據(jù) deleted_flag 變成 1,代表數(shù)據(jù)被刪除。但是數(shù)據(jù)沒有被清空,在新一行數(shù)據(jù)大小小于這一行的時(shí)候,可能會占用這一行。這樣其實(shí)就是存儲碎片。
之后,相關(guān)數(shù)據(jù)的索引需要更新,清除這些數(shù)據(jù)。并且,會產(chǎn)生對應(yīng)的 binlog 與 redolog 日志。
如果 delete 的數(shù)據(jù)是大量的數(shù)據(jù),則會:
我們很容易想到,在 delete 后加上 limit 限制控制其數(shù)量,這個數(shù)量讓他會走索引,從而不會鎖整個表。
但是,存儲碎片,主從同步,占用空間的問題并沒有解決??梢栽趧h除完成后,通過如下語句,重建表:
alter table 你的表 engine=InnoDB, ALGORITHM=INPLACE, LOCK=NONE;
注意這句話其實(shí)就是重建你的表,雖然你的表的引擎已經(jīng)是 innodb 了,加上后面的, ALGORITHM=INPLACE, LOCK=NONE 可以不用鎖表就重建表。
還有一種方案是,新建一張同樣結(jié)構(gòu)的表,在原有表上加上觸發(fā)器:
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;
這樣可以保證線上業(yè)務(wù)有新數(shù)據(jù)會同步。之后,將所有企業(yè)類型的數(shù)據(jù),插入新表,同時(shí)如果已存在則證明發(fā)生了更新同步就不插入。個體戶數(shù)據(jù)由于業(yè)務(wù)變化,并不在這個表上更新,所以這樣通過了無表鎖同步實(shí)現(xiàn)了大表的數(shù)據(jù)清理
到此這篇關(guān)于mysql 大表批量刪除大量數(shù)據(jù)的實(shí)現(xiàn)方法的文章就介紹到這了,更多相關(guān)mysql 大表批量刪除內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
標(biāo)簽:甘南 黑河 吉林 荊州 錦州 隨州 滄州 資陽
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《mysql 大表批量刪除大量數(shù)據(jù)的實(shí)現(xiàn)方法》,本文關(guān)鍵詞 mysql,大表,批量,刪除,大量,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。