首頁 > 軟體

一步步教你如何使用mysql binlog恢復資料

2023-10-09 22:00:21

如果想通過 mysql 的 binlog 恢復資料,首先要開啟 binlog 。這裡搭建一個測試的環境,瞭解一下 mysql binlog 是如何恢復資料庫的。原理比較簡單,binlog 會儲存mysql中變化的資料,比如你建立了一個資料庫,寫入了一些資料,這些都會儲存在 mysql 的 binlog 中。

需要恢復的時候就找到,兩個位置,一個起始位置,一個結束的位置。結束的位置,一半是資料被破壞或者刪除前的位置。mysql 8 預設已經開啟了 binlog

mysql> show variables like '%log_bin%';
+---------------------------------+--------------------------+
| Variable_name                   | Value                    |
+---------------------------------+--------------------------+
| log_bin                         | ON                       |
| log_bin_basename                | /data/mysql/binlog       |
| log_bin_index                   | /data/mysql/binlog.index |
| log_bin_trust_function_creators | OFF                      |
| log_bin_use_v1_row_events       | OFF                      |
| sql_log_bin                     | ON                       |
+---------------------------------+--------------------------+

可以看到 log_bin 已經開啟, 同時可以看到儲存的位置在 /daba/mysql 目錄 , 字首是 binlog

ls /data/mysql/binlog.*
/data/mysql/binlog.000143  /data/mysql/binlog.000144  /data/mysql/binlog.000145  /data/mysql/binlog.000146  /data/mysql/binlog.index

可以看到有好幾個 binlog 紀錄檔檔案, 因為這裡是測試的資料庫(有歷史資訊,之前的資料,沒什麼用),為了方便測試,直接進行重置操作。(刪除了全部binlog檔案) ,如果是生成環境 ,謹慎操作,資料無價。

show master logs;
+---------------+-----------+-----------+
| Log_name      | File_size | Encrypted |
+---------------+-----------+-----------+
| binlog.000143 |       156 | No        |
| binlog.000144 |       200 | No        |
| binlog.000145 |       156 | No        |
| binlog.000146 |       156 | No        |
+---------------+-----------+-----------+
4 rows in set (0.01 sec)

mysql> reset master;
Query OK, 0 rows affected (0.02 sec)

mysql> show master logs;
+---------------+-----------+-----------+
| Log_name      | File_size | Encrypted |
+---------------+-----------+-----------+
| binlog.000001 |       156 | No        |
+---------------+-----------+-----------+
1 row in set (0.00 sec)


ls /data/mysql/binlog.*
/data/mysql/binlog.000001  /data/mysql/binlog.index

執行重置(reset master)後 ,可以看到之前的 binlog 檔案已經被刪除了,產生一個新的 binlog 檔案。

可以檢視一下這個檔案的內容

$mysqlbinlog binlog.000001
# The proper term is pseudo_replica_mode, but we use this compatibility alias
# to make the statement usable on server versions 8.0.24 and older.
/*!50530 SET @@SESSION.PSEUDO_SLAVE_MODE=1*/;
/*!50003 SET @OLD_COMPLETION_TYPE=@@COMPLETION_TYPE,COMPLETION_TYPE=0*/;
DELIMITER /*!*/;
...

這個時候,我們新增一些資料

mysql <<EOT
create database test_liuhaolin_com;
select now();
EOT

mysql <<EOT
use test_liuhaolin_com;

create table if not exists test(
	`id` int unsigned not null auto_increment primary key,
	`key` varchar(100),
	`val` varchar(255)
) engine=myisam charset=utf8mb4;
EOT

mysql <<EOT
use test_liuhaolin_com;
insert into test  values ('1','website', 'https://www.liuhaolin.com');
EOT

這個時候,不小心刪除了,資料庫 test_liuhaolin_com

mysql> drop database test_liuhaolin_com;
Query OK, 1 row affected (0.09 sec)

現在就需要解決一個實際問題,怎麼恢復這個資料庫

首先為了防止干擾,執行 flush logs ,產生一個新binlog 檔案。

mysql> show master logs;
+---------------+-----------+-----------+
| Log_name      | File_size | Encrypted |
+---------------+-----------+-----------+
| binlog.000001 |      1594 | No        |
+---------------+-----------+-----------+
1 row in set (0.00 sec)

mysql> show master status;
+---------------+----------+--------------+------------------+-------------------+
| File          | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+---------------+----------+--------------+------------------+-------------------+
| binlog.000001 |     1594 |              |                  |                   |
+---------------+----------+--------------+------------------+-------------------+
1 row in set (0.00 sec)

mysql> flush logs;
Query OK, 0 rows affected (0.08 sec)

mysql> show master status;
+---------------+----------+--------------+------------------+-------------------+
| File          | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+---------------+----------+--------------+------------------+-------------------+
| binlog.000002 |      156 |              |                  |                   |
+---------------+----------+--------------+------------------+-------------------+
1 row in set (0.00 sec)

恢復資料,首先要找到資料在哪裡被刪除了。

mysqlbinlog binlog.000001 | grep -n  "drop database"
113:drop database test_liuhaolin_com

可以看到在 113 行的地方有個 刪除語句。終可以找到兩個地方

  1. 資料需要恢復的起始位置
  2. 資料需要恢復的結束位置

這裡起始的位置就找 建立資料庫的位置,結束的位置就找 刪除資料庫的位置。

mysqlbinlog --set-charset=utf8  binlog.000001 > tmp.sql

可以在檔案 tmp.sql 中知道 開始和結束位置

binlog

可以看到開始的地方是 233

binlog

可以看到結束的位置在 1371 ,所以執行一下資料的恢復。

mysqlbinlog -v binlog.000001 --start-position=233 --stop-position=1371 | mysql

作為驗證,執行前可以,看下 資料庫是否存在。

mysql> use test_liuhaolin_com;
ERROR 1049 (42000): Unknown database 'test_liuhaolin_com'

# 執行恢復操作
mysqlbinlog -v  binlog.000001  --start-position=233  --stop-position=1371 | mysql 

# 再次檢查,可以發現資料已經恢復
mysql>
mysql> use test_liuhaolin_com;
Database changed

總結

到此這篇關於如何使用mysql binlog 恢復資料的文章就介紹到這了,更多相關mysql binlog恢復資料內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com