本機用的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