mysql恢复之delete 忘加where条件误删除恢复(binglog格式必须是ROW)

(一)恢复的前提条件就是开启了二进制日志和格式为行格式,二个条件缺一不行!

mysql> show variables like "%log_bin%";

+---------------------------------+---------------------------------------+
| Variable_name                   | Value                                 |
+---------------------------------+---------------------------------------+
| log_bin                         | ON                                    |
| log_bin_basename                | /usr/local/mysql/data/mysql_bin       |
| log_bin_index                   | /usr/local/mysql/data/mysql_bin.index |
| log_bin_trust_function_creators | OFF                                   |
| log_bin_use_v1_row_events       | OFF                                   |
| sql_log_bin                     | ON                                    |

+---------------------------------+---------------------------------------+


mysql> show variables like "%binlog_format%";
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| binlog_format | ROW   |
+---------------+-------+


(二)准备测试数据

mysql> create  table testdb.person_t(id int,name varchar(10),address varchar(10));
Query OK, 0 rows affected (0.03 sec)


mysql> insert into testdb.person_t values(1,'anzhen','nanning'),(2,'liming','beijng'),(3,'xiaohon','gzhou'),(4,'sfds','nhui');
Query OK, 4 rows affected (0.05 sec)
Records: 4  Duplicates: 0  Warnings: 0

mysql> select * from testdb.person_t;
+------+---------+---------+
| id   | name    | address |
+------+---------+---------+
|    1 | anzhen  | nanning |
|    2 | liming  | beijng  |
|    3 | xiaohon | gzhou   |
|    4 | sfds    | nhui    |
+------+---------+---------+
4 rows in set (0.00 sec)


(三)误delete操作,没有加where条件

mysql> delete from testdb.person_t;
Query OK, 4 rows affected (0.01 sec)


mysql> commit;
Query OK, 0 rows affected (0.00 sec)


mysql> select * from testdb.person_t;
Empty set (0.00 sec)

(四)首先要确定delete误操作写进了那个binlog二进制日志文件里面,然后这个二进制日志从中恢复相关行操作。因为启用的是row行格式,所有的每条记录

的修改等等操作都会记录在二进制日志里面。

如果没有其他人flush logs的话,误操作就记录在mysql_bin.000401这个二进制日志文件中。如果有他人flush logs。就往以上找。直到发现误操作的记录。

开始恢复,在线上的话,应该比较复杂,要先进行锁表,以免数据再次被污染。(锁表,查看正在写哪个二进制日志)

mysql> lock table testdb.person_t read;
Query OK, 0 rows affected (0.00 sec)

mysql> show master status;
+------------------+----------+--------------+------------------+-------------------+
| File             | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+------------------+----------+--------------+------------------+-------------------+
| mysql_bin.000401 |     1045 |              |                  |                   |
+------------------+----------+--------------+------------------+-------------------+
1 row in set (0.00 sec)

mysql> flush logs;

(五)查看binlog的相关信息 

查看MySQL的安装目录:

mysql> show variables like "%basedir%";
+---------------+-------------------+
| Variable_name | Value             |
+---------------+-------------------+
| basedir       | /usr/local/mysql/ |
+---------------+-------------------+
1 row in set (0.00 sec)

查看binlog的日志目录和格式:

mysql> show variables like "%log_bin%";
+---------------------------------+---------------------------------------+
| Variable_name                   | Value                                 |
+---------------------------------+---------------------------------------+
| log_bin                         | ON                                    |
| log_bin_basename                | /usr/local/mysql/data/mysql_bin       |
| log_bin_index                   | /usr/local/mysql/data/mysql_bin.index |
| log_bin_trust_function_creators | OFF                                   |
| log_bin_use_v1_row_events       | OFF                                   |
| sql_log_bin                     | ON                                    |
+---------------------------------+---------------------------------------+
6 rows in set (0.01 sec)


(六)新建一个session。查看mysqlbinlog里面delete操作的格式,因为版本不一样,可能格式不一样。

/usr/local/mysql/bin/mysqlbinlog --no-defaults --base64-output=decode-rows  -v -v /usr/local/mysql/data/mysql_bin.000401 > /tmp/info.txt

[root@localhost104 mysql]# cat /tmp/info.txt | grep delete
[root@localhost104 mysql]# cat /tmp/info.txt | grep DELETE
### DELETE FROM `testdb`.`person_t`
### DELETE FROM `testdb`.`person_t`
### DELETE FROM `testdb`.`person_t`
### DELETE FROM `testdb`.`person_t`


(七)在binglog中去查找相关误delete操作的记录,delete from testdb.person_t操作删除了四行数据mysql_bin.000401,建议写全部目录。

[root@localhost104 mysql]# /usr/local/mysql/bin/mysqlbinlog --no-defaults --base64-output=decode-rows  -v -v /usr/local/mysql/data/mysql_bin.000401| sed -n '/### DELETE FROM `testdb`.`person_t`/,/COMMIT/p' > /tmp/delete.txt 
[root@localhost104 mysql]# 
[root@localhost104 mysql]# 
[root@localhost104 mysql]# cat  /tmp/delete.txt
### DELETE FROM `testdb`.`person_t`
### WHERE
###   @1=1 /* INT meta=0 nullable=1 is_null=0 */
###   @2='anzhen' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
###   @3='nanning' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
### DELETE FROM `testdb`.`person_t`
### WHERE
###   @1=2 /* INT meta=0 nullable=1 is_null=0 */
###   @2='liming' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
###   @3='beijng' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
### DELETE FROM `testdb`.`person_t`
### WHERE
###   @1=3 /* INT meta=0 nullable=1 is_null=0 */
###   @2='xiaohon' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
###   @3='gzhou' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
### DELETE FROM `testdb`.`person_t`
### WHERE
###   @1=4 /* INT meta=0 nullable=1 is_null=0 */
###   @2='sfds' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
###   @3='nhui' /* VARSTRING(10) meta=10 nullable=1 is_null=0 */
# at 1014
#170723 11:35:54 server id 10  end_log_pos 1045 CRC32 0x731dc552        Xid = 113
COMMIT/*!*/;

(8)把delete操作还原成insert操作。这样可以把数据找回来。@1,@2,@3分别表示第1个字段,第2个字段,第3个字段。误删的表有3个字段,所

以就配置cat /tmp/delete.txt | sed -n '/###/p' | sed 's/### //g;s/\/\*.*/,/g;s/DELETE FROM/INSERT INTO/g;s/WHERE/SELECT/g;' | sed -r 's/(@3.*),/\1;/g' | sed 's/@[1-3]=//g' > /tmp/person_t.sql,如果有17个字段,就配置为cat /tmp/delete.txt | 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-17]=//g' > /tmp/person_t.sql就可以了。只需要改数字。

 

[root@localhost104 mysql]# cat /tmp/delete.txt | sed -n '/###/p' | sed 's/### //g;s/\/\*.*/,/g;s/DELETE FROM/INSERT INTO/g;s/WHERE/SELECT/g;' | sed -r 's/(@3.*),/\1;/g' | sed 's/@[1-3]=//g' > /tmp/person_t.sql

[root@localhost104 mysql]# cat   /tmp/person_t.sql
INSERT INTO `testdb`.`person_t`
SELECT
  1 ,
  'anzhen' ,
  'nanning' ;
INSERT INTO `testdb`.`person_t`
SELECT
  2 ,
  'liming' ,
  'beijng' ;
INSERT INTO `testdb`.`person_t`
SELECT
  3 ,
  'xiaohon' ,
  'gzhou' ;
INSERT INTO `testdb`.`person_t`
SELECT
  4 ,
  'sfds' ,
  'nhui' ;

查看一下数据是否是要恢复的数据,如果是的话,就可以进行恢复操作。

在进行lock表的mysql客户端里面执行。

 mysql>unlock tables;  source /tmp/person_t.sql;

Query OK, 1 row affected (0.03 sec)
Records: 1  Duplicates: 0  Warnings: 0


Query OK, 1 row affected (0.02 sec)
Records: 1  Duplicates: 0  Warnings: 0


Query OK, 1 row affected (0.01 sec)
Records: 1  Duplicates: 0  Warnings: 0


Query OK, 1 row affected (0.01 sec)
Records: 1  Duplicates: 0  Warnings: 0


(9)验证数据是否恢复完成!

mysql> select * from testdb.person_t;
+------+---------+---------+
| id   | name    | address |
+------+---------+---------+
|    1 | anzhen  | nanning |
|    2 | liming  | beijng  |
|    3 | xiaohon | gzhou   |
|    4 | sfds    | nhui    |
+------+---------+---------+
5 rows in set (0.00 sec)

恢复成功!

可以看见数据已经完全恢复,这种方法的优点是快速,方便。前提是在误操作之前开启了二进制日志和行格式。

到这里数据就完整回来了。将binglog格式设置为row有利有弊,好处是记录了每一行的实际变化,在主从复制时也不容易出问题。但是由于记录每行的变化,会占用大量磁盘,主从复制时带宽占用会有所消耗。到底是使用row还是mixed,需要在实际工作中自己去衡量,但从整体上来说,binglog的格式设置为row,都是不二的选择。

总结:

所以在数据库操作的过程中我们需要格外小心,当然开发那边我们需要做好权限的控制,不过有一个参数可以解决我们的问题,让我们不用担心类似的问题发生:

在[mysql]段落开启这个参数:

safe-updates

这样当我们在做DML操作时忘记加where条件时,mysqld服务器是不会执行操作的:

复制代码
mysql> select *  from t1;
+----+------------------+
| id | name             |
+----+------------------+
|  1 | yayun            |
|  2 | atlas            |
|  3 | mysql            |
|  6 | good yayun heheh |
+----+------------------+
4 rows in set (0.00 sec)

mysql> delete from t1;
ERROR 1175 (HY000): You are using safe update mode and you tried to update a table without a WHERE that uses a KEY column
mysql> 


<think>好的,我现在需要帮助用户了解如何在MySQL恢复DELETE删除的数据。首先,我得回忆一下相关知识。记得MySQLbinlog可以用于数据恢复,但具体步骤可能有点复杂。用户提供的引用里提到了一些关键点,比如使用mysqlbinlog工具、查找误操作的时间点、转换DELETE语句为INSERT等。我需要把这些步骤整理清楚,确保逻辑正确,同时遵循用户给出的格式要求。 首先,恢复数据的前提是启用了binlog,并且日志格式ROW模式。因为STATEMENT或MIXED模式可能无法完整记录数据变化。用户可能不知道这个前提,需要明确说明。 接下来,步骤大概分为定位误操作的时间点、导出相关binlog内容、转换DELETE为INSERT语句,然后执行这些语句恢复数据。每个步骤需要详细说明,比如如何用mysqlbinlog命令指定时间范围,如何用sed命令处理日志文件。用户提供的引用里提到了具体的命令示例,应该参考这些内容,确保准确性。 还要注意可能的错误点,比如时间点不准确导致导出内容不全,或者转换过程中语法错误。需要提醒用户在操作前备份当前数据,避免二次破坏。此外,如果有多个事务或大量数据,处理起来可能更复杂,可能需要分段处理或使用工具辅助。 用户的问题可能还涉及权限问题,比如是否有权限访问binlog文件,或者执行mysqlbinlog命令。不过这可能属于进阶内容,用户如果没有提到,可能不需要深入,但可以简单提及需要注意权限。 最后,生成相关问题时要围绕数据恢复的主题,比如其他恢复方法、预防措施、binlog的其他用途等,这样用户可以有进一步的学习方向。 现在需要将以上思考整理成结构清晰的步骤,使用中文,注意数学公式和代码块的格式要求,同时正确引用提供的参考资料。确保每个步骤都有对应的引用标注,例如使用[^1]、[^2]等。还要检查LaTeX语法是否正确,行内公式用$...$,独立公式用$$...$$,但在这个问题中可能公式较少,主要关注代码块和引用。</think>### MySQL恢复DELETE删除数据的方法 在MySQL恢复被`DELETE`误删的数据,主要依赖**binlog日志**和**ROW格式的日志记录**。以下是详细步骤: --- #### **一、前提条件** 1. **启用binlog**:MySQL需配置`log_bin=ON`,并设置`binlog_format=ROW`[^4]。 2. **定位误操作时间**:需知道`DELETE`执行的大致时间范围[^4]。 --- #### **二、恢复流程** 1. **查找误操作的binlog点位** 使用`mysqlbinlog`工具导出特定时间段的日志: ```bash mysqlbinlog "/path/mysql-bin.00000X" \ --start-datetime="2024-09-22 22:00:00" \ --stop-datetime="2024-09-22 22:24:00" \ --base64-output=decode-rows -v > operation.sql ``` 通过搜索`DELETE FROM`关键字定位误删记录[^4][^5]。 2. **转换DELETE语句为INSERT** 使用`sed`命令将`DELETE`操作逆向转换为`INSERT`语句: ```bash cat operation.sql | sed -n '/###/p' | \ sed 's/### //g; s/DELETE FROM/INSERT INTO/g; s/WHERE/SELECT/g;' \ | sed 's/@1=//g; s/@[1-9]=/,/g' > restore.sql ``` 此操作会将删除前的数据生成插入语句。 3. **执行恢复脚本** 在MySQL中运行生成的`restore.sql`文件: ```bash mysql -u root -p < restore.sql ``` 或通过客户端工具(如Navicat)逐条执行[^3]。 --- #### **三、注意事项** - **日志格式**:必须为`ROW`格式,否则无法解析具体数据[^1]。 - **及时性**:误操作后尽快恢复,避免binlog被覆盖或清理。 - **备份优先**:恢复前备份当前数据库状态,防止二次损坏。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值