一句DELETE引發的加班(Mysql 恢復Delete刪除的數據)


 本機用的Navicat連mysql測試DB又連了正式DB,因為本地與正式要頻繁操作所以都打開了很多查詢,本來要DELETE刪除測試DB的數據,沒看清在正式環境執行了。共刪除了325條數據,然后在網上找恢復數據的辦法,一定要是DELETE刪除的,如果用的是drop table刪除表是沒辦法恢復的,本次操作是Windows,如果是Liunx下的Mysql過程流程完全一樣,也可以胡考的,具體恢復流程如下

第一步:先查看binlog功能是否開啟

show variables like '%log_bin%';

如果為log_bin為ON說明可以恢復,如果為OFF說明沒有開啟binlog,也沒有預先生成回滾SQL,那可能真的無法快速回滾了,GAMEOVER,下面的不用看了(好像可以通過ibd恢復,具體可以參考 https://blog.csdn.net/hanjun0612/article/details/102466509 )。

   第二步:查看數據文件存方路徑

show variables like '%datadir%';

 打開數據庫所在路徑查看有mysql-bin.****這樣的文件,注意DELETE刪除的時間,對比mysql-bin文件的修改時間,我的是26號下午18點左右進行的刪除數據,所以找mysql-bin.000028這個文件

 

  第三步:找到mysql安裝目錄

show variables like "%basedir%";

 CMD命令符進行Mysql安裝目錄下,找到mysqlbinlog.exe,如果沒有說明你安裝的是假mysql.

 

  第四步.通過mysqlbinlog 恢復刪除的數據日志記錄

mysqlbinlog --base64-output=decode-rows -v --database=DBName --start-datetime="2019-11-26 18:00:00" --stop-datetime="2019-11-26 18:10:00" D:\MySQL\Data\mysql-bin.000028 > mysqllog.sql
mysqlbinlog 命令的參數說明
--base64-output=decode-rows //數據轉換正常的字符,如果不設置這個參數將顯示base64的數據
--database=DBName  //數據庫名(一個mysql數據庫比較多,指定方便恢復)
--start-datetime="2019-11-26 18:00:00"  //恢復起始時間
--stop-datetime="2019-11-26 18:10:00"  //恢復結束時間
D:\MySQL\Data\mysql-bin.000028  //為數據恢復的日志文件
mysqllog.sql    //恢復以后我們需要的文件名

 執行以后如下圖:

 

 

 打開mysqllog.sql文件,搜索 DELETE 關鍵字,找到被刪除數據,看到這些數據以后總算放心了,如下圖:

  第五步:把mysqllog的DELETE轉換為Insert語句,這個在liunx下操作方便(有人用python轉換也可以),把文件mysqllog.sql復制Liunx下

cat mysqllog.sql | sed -n '/###/p' | sed 's/### //g;s/\/\*.*/,/g;s/DELETE FROM/;INSERT INTO/g;s/WHERE/SELECT/g;' |sed -r 's/(@17.*),/\1;/g' | sed 's/@1=//g'| sed 's/@[1-9]=/,/g' | sed 's/@[1-9][0-9]=/,/g' > mysqllogOK.sql

 

 轉換以后打開mysqllogOK.sql文件,因為批量替換前面多個一個分號,去掉以后,把所有生成的Insert語句在navicat下執行即可恢復所有DELETE刪除的數據。

如果沒有liunx可以在windows下用vbs來實現,把下面代碼復制保存為:delete2insert.vbs 文件(一定要是.vbs格式文件) 與mysqllog.sql在同一目錄下,然后雙擊運行,會生成mysqllogOK.sql文件就是我們要的INSERT語句

'========================== 
'用VBS實現 MYSQL binglog DELETE轉INSERT 
'========================== 
function replaceregex(patern,str,tagstr) 
    dim regex,matches 
    set regex=new regExp 
    regex.pattern=patern 
    regex.IgnoreCase=true 
    regex.global=true 
    matches=regex.replace(str,tagstr) 
    replaceregex=matches 
end function

'======Mysql binlog DELETE轉INSERT================
'VBS打開文本文件
Set oldStream = CreateObject("ADODB.Stream")
oldStream.CharSet = "utf-8"
oldStream.Open
oldStream.LoadFromFile("mysqllog.sql") 'binLog生成的DELETE原日志文件
oldText = oldStream.ReadText()
    newText=replace(oldText,"### DELETE FROM", ";INSERT INTO")
    newText=replace(newText,"### WHERE", "SELECT")
    newText=replace(newText,"###", "")
    newText=replace(newText,"@1=", "")
    newText=replaceregex("\@[1-9]=",newText, ",")
    newText=replaceregex("\@[1-9][0-9]=",newText, ",")
oldStream.Close
'VBS保存文件
Set newStream = CreateObject("ADODB.Stream")
newStream.Type = 2 'Specify stream type - we want To save text/string data.
newStream.Charset = "utf-8" 'Specify charset For the source text data.
newStream.Open 'Open the stream And write binary data To the object
newStream.WriteText newText
newStream.SaveToFile "mysqllogOK.sql", 2 'DELETE轉成INSERT以后的新的SQL文件名
newStream.Close

 

 總結:   

1.不要隨便用DELETE,數據做用字段做刪除標記,不用使用DELETE刪除數據,如果非要刪除,刪除前一定先備份一下數據

2.數據庫做好定時備份,如果數據庫不大,可以一小時備份一次,再大也可以每天備份一次

3.做事細心,越熟練的事越要細心,本次DELETE導致的加班,就是以為自己對sql熟練就隨意操作。

4.也是最重要的一點,一定要Mysql開啟binlog功能

如果沒開啟binlog功能的趕緊開啟一下吧,mysql開啟binLog功能方法(windows),打開my.ini文件,添加如下配置,重啟mysql即可開啟

# log-bin
log-bin=mysql-bin binlog_format = ROW


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM