MySQL回滚到某一时刻数据的方法
MySQL回滚到某一时刻数据的方法
Oracle Database 12c RMAN全量+增量备份+归档日志恢复详解
Oracle 12c数据库定时备份和清理脚本
mysql不同于oracle和db2这种企业级数据库,它没有oracle里面的redo日志,也没有db2里面的循环日志。mysql有类似于oracle和db2归档日志的binlog,而这个binlog可以看作是循环日志和归档日志的结合。有一定的大小限制。未完成的事务和已完成的事务都会记录在binlog中,当一个binlog写满之后,就会开启一个新的binlog。binlog还有三种方式,row,statement,mixed。其中row记录量最大,但是对于各种工具支持最好,所以对于安全要求比较高的数据库,推荐使用row格式。
同时,binlog也是mysql主从复制的依据,所以使用binlog来恢复数据库是比较可靠的。不足的就是mysql并没有内置binlog的清理工具,对于长时间的binlog我们需要去手动清理或者编写脚本清理。mysql也没有提供oracle,db2那样的增量备份方法。所以保证binlog不要丢失就比较重要。虽然手动的操作多了一些,但是这也代表着mysql的恢复更偏向于无状态的,即异地跨平台恢复会比较方便,不需要像oracle那样必须找到控制文件。
数据库恢复的过程于oracle,db2区别不大。基本都是通过先恢复全备份,再逐个恢复增量备份,再根据归档日志逐条重做事务,一直重做到你需要恢复到的日期为止。mysql由于没有增量备份,所以先恢复全备,再手动找到binlog中全备时间的那一行,从那一行往后开始执行重做事务,直到你需要的停止的那一行。
下面来介绍一下如何将mysql数据库回滚到某一时刻。大概有如下步骤
1、找一个现有的mysql数据库,先不打开binlog,插入几条数据
2、打开binlog,重启数据库,再插入几条数据
3、使用mysqldump全备一次数据库
4、再插入几条数据,模拟全备之后执行成功的事务,记录执行完毕的时间。
5、模拟数据库崩溃或者误删操作,然后将全备文件和binlog都拷贝到另一台服务器上进行异地恢复。
这里使用的版本是mysql 5.7.18
首先我们先建库建表,但是此时没有开启binlog,这里我主要像说明,如果没有binlog,那么就没有归档日志,我们就不知道以前做过了哪些事务,只能使用全备进行恢复,而全备之后发生的操作就都会丢失。my.cnf配置如下
- #server-id = 1
- #log_bin = /var/log/mysql/mysql-bin.log
- expire_logs_days = 10
- max_binlog_size = 100M
可以看见log_bin前面被注释掉了,也就是没有开启。下面开始建库建表
- mysql> show databases;
- +--------------------+
- | Database |
- +--------------------+
- | information_schema |
- | mydb |
- | mysql |
- | performance_schema |
- | sys |
- +--------------------+
- 5 rows in set (0.00 sec)
- mysql> create database dbtest;
- Query OK, 1 row affected (0.00 sec)
- mysql> use dbtest;
- Database changed
- mysql> create table table1 ( id int primary key, name varchar(40), birthday datetime);
- Query OK, 0 rows affected (0.44 sec)
- mysql> insert into table1 values (1,'befor_binlog1',NOW());
- Query OK, 1 row affected (0.05 sec)
- mysql> insert into table1 values (2,'befor_binlog2',NOW());
- Query OK, 1 row affected (0.07 sec)
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- +----+---------------+---------------------+
- 2 rows in set (0.00 sec)
此时我们是看不到当前binlog是哪个文件第几行的
- mysql> show master status \G
- Empty set (0.00 sec)
可以发现,再不开启binlog的情况下,是可以正常插入数据的,但我还是推荐从开始,也就是建库建表之前就开启binlog,那样即使没有全备,也可以从头开始恢复。但是现在这种情况下,如果出现了误删操作,我们是无法拯救我们的数据的。
然后我们执行第二步,开启binlog然后重启数据库。首先修改配置文件my.cnf
- server-id = 1
- log_bin = /var/log/mysql/mysql-bin.log
- expire_logs_days = 10
- max_binlog_size = 100M
然后我们再来插入两条数据
- mysql> use dbtest;
- Reading table information for completion of table and column names
- You can turn off this feature to get a quicker startup with -A
- Database changed
- mysql> insert into table1 values (3,'after_binlog1',NOW());
- Query OK, 1 row affected (0.12 sec)
- mysql> insert into table1 values (4,'after_binlog2',NOW());
- Query OK, 1 row affected (0.20 sec)
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- +----+---------------+---------------------+
- 4 rows in set (0.00 sec)
如果此时发生了误删,那我们在进行异地恢复的时候,只能恢复出id为3和4的两条数据,因为1,2在插入的时候没有开启binlog,binlog中没有这两条事务的记录,所以就恢复不了。
下面我们进行第三部,使用mysqldump进行一次全备
- root@f4d417a2e6ea:/# mysqldump --single-transaction --master-data=2 --triggers --routines --all-databases -uroot -p > /backup/full.sql
- Enter password:
查看一下备份出来的文件所在时刻归档日志的为止
- --
- -- Position to start replication or point-in-time recovery from
- --
- -- CHANGE MASTER TO MASTER_LOG_FILE='mysql-bin.000001', MASTER_LOG_POS=740;
此时查看一下数据库里归档日志的位置
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000001
- Position: 740
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
可以看到,在备份前和备份后,归档日志并没有发生变化,还是停留在同一行里。这个行数和文件名被记录在了全备文件中,以后会用到。
现在如果出现了误删或者存储损坏我们是可以高枕无忧的,因为有了全备,我们可以轻松的异地恢复回来
我们进行第四步,插入两条新数据
- mysql> insert into table1 values (5,'after_backup',NOW());
- Query OK, 1 row affected (0.13 sec)
- mysql> insert into table1 values (6,'after_backup2',NOW());
- Query OK, 1 row affected (0.11 sec)
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- | 5 | after_backup | 2018-06-09 09:11:18 |
- | 6 | after_backup2 | 2018-06-09 09:11:25 |
- +----+---------------+---------------------+
- 6 rows in set (0.00 sec)
然后我们进行第六步,假装系统崩溃存储损坏,我们来尝试恢复到当前数据。在这里我们分两步来做一个是系统的崩溃,我们需要异地恢复到最新为止,另一个是进行了误删操作,我们将6条数据全部恢复回来。
1、系统崩溃情况下,我们先将全备文件和binlog都拷贝到要恢复的新服务器上,这里要注意binlog可能不止一个,但是本例由于数据量少只有一个binlog。binlog的目录就写在my.cnf我们刚改过的文件里
按照正常的步骤,我们应该先恢复全备,然后根据binlog重做已经提交的事务,在恢复之前,可以看到是没有我们原来的库的
- mysql> show databases;
- +--------------------+
- | Database |
- +--------------------+
- | information_schema |
- | mydb |
- | mysql |
- | performance_schema |
- | sys |
- +--------------------+
- 5 rows in set (0.00 sec)
- mysql>
全备份恢复,恢复其实就是让mysql执行一大段sql,而这段sql就是我们用mysqldump导出来的那个。此时可以不开启新数据库的binlog。开了反而还会变慢
# mysql -uroot -p < /import/full.sql
然后查看一下导入了哪些数据
- mysql> show databases;
- +--------------------+
- | Database |
- +--------------------+
- | information_schema |
- | dbtest |
- | mydb |
- | mysql |
- | performance_schema |
- | sys |
- +--------------------+
- 6 rows in set (0.00 sec)
- mysql> use dbtest;
- Reading table information for completion of table and column names
- You can turn off this feature to get a quicker startup with -A
- Database changed
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- +----+---------------+---------------------+
- 4 rows in set (0.00 sec)
不出所料,由于全备时间点的关系,只有id为1-4的被恢复回来了,5,6由于是全备之后插入的,所以当前没有,我们需要从binlog中恢复。这里我们使用mysql自带的mysqlbinlog工具,将binlog解析成sql,然后用mysql执行这段sql,就把后面的事务给执行了,
- root@5d41cbeb4b4d:/import/mysql# ls
- error.log mysql-bin.000001 mysql-bin.index
- root@5d41cbeb4b4d:/import/mysql# mysqlbinlog --no-defaults /import/mysql/mysql-bin.000001 | mysql -uroot -p
- Enter password:
- ERROR 1062 (23000) at line 38: Duplicate entry '3' for key 'PRIMARY'
- mysql> select * from table1
- -> ;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- +----+---------------+---------------------+
- 4 rows in set (0.00 sec)
我们会发现报了一个错,同时数据也没有导入成功。这是什么原因呢。
由于我们在id=2之后,开启了binlog,所以此时binlog中的内容就是insert id为3,4,5,6的语句,所以当我们把整个binlog文件全部重做一遍的话,在insert id=3的时候就会报主键冲突的错误,这点是显而易见的。那么如何避免重复执行已备份的事务呢,这就要我们手动指定binlog的重做时间点。前面我们已经知道,从全备文件full.sql中可以看到备份时间点的binlog文件和行数,也就是mysql-bin.000001的第740行,所以我们就从这一行开始恢复
- # mysqlbinlog --no-defaults --start-position=740 /import/mysql/mysql-bin.000001 | mysql -
- uroot -p
- Enter password:
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- | 5 | after_backup | 2018-06-09 09:11:18 |
- | 6 | after_backup2 | 2018-06-09 09:11:25 |
- +----+---------------+---------------------+
- 6 rows in set (0.00 sec)
现在我们发现已经成功的恢复了全部的6条数据
2、上面是针对系统崩溃这种情况,进行的全量恢复操作,那么如果是因为操作事务删除了某些数据或者插入了某些脏数据怎么办呢。
首先我们去源库中删除一些数据,然后插入一些错误数据
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- | 5 | after_backup | 2018-06-09 09:11:18 |
- | 6 | after_backup2 | 2018-06-09 09:11:25 |
- +----+---------------+---------------------+
- 6 rows in set (0.00 sec)
- mysql> delete from table1 where id >3;
- Query OK, 3 rows affected (0.14 sec)
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- +----+---------------+---------------------+
- 3 rows in set (0.00 sec)
- mysql> insert into table1 values (4,'xxxxxxx1',NOW());
- Query OK, 1 row affected (0.12 sec)
- mysql> insert into table1 values (5,'xxxxxxx1',NOW());
- Query OK, 1 row affected (0.14 sec)
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | xxxxxxx1 | 2018-06-09 09:43:00 |
- | 5 | xxxxxxx1 | 2018-06-09 09:43:05 |
- +----+---------------+---------------------+
- 5 rows in set (0.00 sec)
如上所示,先删掉了一半的数据,然后又插入了两条数据,占用了原来的4,5.id=6的数据没了。也就是两条数据被篡改,1条丢失的情况。
此时再看一下binlog写到什么位置了。
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000001
- Position: 2233
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
如果再按照上面的方法对binlog进行恢复,那么我们的误删操作也会被恢复,就失去了意义,所以此时我们必须指定恢复的时间点。
从上面可以看出,进行误删操作是9:11:25之后,所以我们只需恢复到这个时间点就可以了。
首先还是根据前面的全备进行恢复
- mysql> show databases;
- +--------------------+
- | Database |
- +--------------------+
- | information_schema |
- | mydb |
- | mysql |
- | performance_schema |
- | sys |
- +--------------------+
- 5 rows in set (0.00 sec)
- # mysql -uroot -p < /import/full.sql
- Enter password:
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- +----+---------------+---------------------+
- 4 rows in set (0.00 sec)
然后关键点就是既要指定清楚binlog起始位置,又要指明结束时间,这里时间使用UTC时间,要和binlog里面对应注意,id=6的数据是9:11:25才插入成功的,所以我们恢复最早只能是9:11:26,如果恢复到9:11:25是看不到id=6的数据的
# mysqlbinlog --no-defaults --start-position=740 --stop-datetime="2018-06-09 01:11:26" /import/mysql/mysql-bin.000001 | mysql -uroot -p
然后发现我们的恢复成功了
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- | 5 | after_backup | 2018-06-09 09:11:18 |
- | 6 | after_backup2 | 2018-06-09 09:11:25 |
- +----+---------------+---------------------+
- 6 rows in set (0.00 sec)
如果不指明--stop-datetime 那么就会恢复到binlog的末尾,我们的误操作也就一起恢复回去了。
题外话:
刚才看到--stop-datetime要用UTC时间,是因为我们查看binlog得来的,下面我们就来看看如何查看binlog文件
比如我们可以看到以下binlog的记录
- mysqlbinlog --no-defaults /import/mysql/mysql-bin.000001
- ......
- # at 1179
- #180609 1:11:25 server id 1 end_log_pos 1235 CRC32 0x6b867c5b Table_map: `dbtest`.`table1` mapped to number 254
- # at 1235
- #180609 1:11:25 server id 1 end_log_pos 1294 CRC32 0xc058c50f Write_rows: table id 254 flags: STMT_END_F
- BINLOG '
- PSkbWxMBAAAAOAAAANMEAAAAAP4AAAAAAAEABmRidGVzdAAGdGFibGUxAAMDDxIDeAAABlt8hms=
- PSkbWx4BAAAAOwAAAA4FAAAAAP4AAAAAAAEAAgAD//gGAAAADWFmdGVyX2JhY2t1cDKZoBIS2Q/F
- WMA=
- '/*!*/;
- # at 1294
- #180609 1:11:25 server id 1 end_log_pos 1325 CRC32 0x8113be5c Xid = 992
- COMMIT/*!*/;
- # at 1325
- #180609 1:42:32 server id 1 end_log_pos 1390 CRC32 0x6c1bff71 Anonymous_GTID last_committed=4 sequence_number=5
- SET @@SESSION.GTID_NEXT= 'ANONYMOUS'/*!*/;
- # at 1390
- #180609 1:42:32 server id 1 end_log_pos 1464 CRC32 0xfb668ab1 Query thread_id=11 exec_time=0 error_code=0
- SET TIMESTAMP=1528508552/*!*/;
- BEGIN
- /*!*/;
- # at 1464
- #180609 1:42:32 server id 1 end_log_pos 1520 CRC32 0x89d622f5 Table_map: `dbtest`.`table1` mapped to number 254
- # at 1520
- #180609 1:42:32 server id 1 end_log_pos 1626 CRC32 0x4c8f4330 Delete_rows: table id 254 flags: STMT_END_F
- BINLOG '
- iDAbWxMBAAAAOAAAAPAFAAAAAP4AAAAAAAEABmRidGVzdAAGdGFibGUxAAMDDxIDeAAABvUi1ok=
- iDAbWyABAAAAagAAAFoGAAAAAP4AAAAAAAEAAgAD//gEAAAADWFmdGVyX2JpbmxvZzKZoBILJvgF
- AAAADGFmdGVyX2JhY2t1cJmgEhLS+AYAAAANYWZ0ZXJfYmFja3VwMpmgEhLZMEOPTA==
- '/*!*/;
上面展示了我们最后一次正常的插入操作和第一次误删操作的日志。可以看到误删操作是1325开始,1520结束,1:42:32秒一秒钟之内结束。而最后一次正常插入是1235行结束,1:11:25时候。
然后我们再找全备起始时间也就是740行所对应的时间
- # at 709
- #180609 0:44:38 server id 1 end_log_pos 740 CRC32 0x1119b274 Xid = 15
- COMMIT/*!*/;
- # at 740
- #180609 1:11:18 server id 1 end_log_pos 805 CRC32 0x7732118b Anonymous_GTID last_committed=2 sequence_number=3
- SET @@SESSION.GTID_NEXT= 'ANONYMOUS'/*!*/;
- # at 805
- #180609 1:11:18 server id 1 end_log_pos 887 CRC32 0x22d66d83 Query thread_id=10 exec_time=0 error_code=0
- SET TIMESTAMP=1528506678/*!*/;
- BEGIN
- /*!*/;
可以看到740行所对应的是1:11:18
但是奇怪的事情是,如果我们按照上面的,--start-position和s--stop-datetime相结合,那么binlog并不是从740行1294行,而是从第四行开始的,如下
- # mysqlbinlog --no-defaults --start-position=740 --stop-datetime="2018-06-09 01:11:26" /import/mysql/mysql-bin.000001
- /*!50530 SET @@SESSION.PSEUDO_SLAVE_MODE=1*/;
- /*!50003 SET @OLD_COMPLETION_TYPE=@@COMPLETION_TYPE,COMPLETION_TYPE=0*/;
- DELIMITER /*!*/;
- # at 4
- #180609 0:43:43 server id 1 end_log_pos 123 CRC32 0x73c53756 Start: binlog v 4, server v 5.7.18-0ubuntu0.16.04.1-log created 180609 0:43:43 at startup
- # Warning: this binlog is either in use or was not closed properly.
- ROLLBACK/*!*/;
- BINLOG '
- vyIbWw8BAAAAdwAAAHsAAAABAAQANS43LjE4LTB1YnVudHUwLjE2LjA0LjEtbG9nAAAAAAAAAAAA
- AAAAAAAAAAAAAAAAAAC/IhtbEzgNAAgAEgAEBAQEEgAAXwAEGggAAAAICAgCAAAACgoKKioAEjQA
- AVY3xXM=
- '/*!*/;
- # at 740
- #180609 1:11:18 server id 1 end_log_pos 805 CRC32 0x7732118b Anonymous_GTID last_committed=2 sequence_number=3
- SET @@SESSION.GTID_NEXT= 'ANONYMOUS'/*!*/;
- # at 805
- #180609 1:11:18 server id 1 end_log_pos 887 CRC32 0x22d66d83 Query thread_id=10 exec_time=0 error_code=0
- SET TIMESTAMP=1528506678/*!*/;
- SET @@session.pseudo_thread_id=10/*!*/;
- SET @@session.foreign_key_checks=1, @@session.sql_auto_is_null=0, @@session.unique_checks=1, @@session.autocommit=1/*!*/;
- SET @@session.sql_mode=1436549152/*!*/;
- SET @@session.auto_increment_increment=1, @@session.auto_increment_offset=1/*!*/;
- /*!\C utf8 *//*!*/;
- SET @@session.character_set_client=33,@@session.collation_connection=33,@@session.collation_server=33/*!*/;
- SET @@session.time_zone='SYSTEM'/*!*/;
- SET @@session.lc_time_names=0/*!*/;
- SET @@session.collation_database=DEFAULT/*!*/;
- BEGIN
- /*!*/;
- # at 887
- #180609 1:11:18 server id 1 end_log_pos 943 CRC32 0x5d815b86 Table_map: `dbtest`.`table1` mapped to number 254
- # at 943
- #180609 1:11:18 server id 1 end_log_pos 1001 CRC32 0x5f1763c3 Write_rows: table id 254 flags: STMT_END_F
- BINLOG '
- NikbWxMBAAAAOAAAAK8DAAAAAP4AAAAAAAEABmRidGVzdAAGdGFibGUxAAMDDxIDeAAABoZbgV0=
- NikbWx4BAAAAOgAAAOkDAAAAAP4AAAAAAAEAAgAD//gFAAAADGFmdGVyX2JhY2t1cJmgEhLSw2MX
- Xw==
- '/*!*/;
- # at 1001
- #180609 1:11:18 server id 1 end_log_pos 1032 CRC32 0x38b9cbf4 Xid = 991
- COMMIT/*!*/;
- # at 1032
- #180609 1:11:25 server id 1 end_log_pos 1097 CRC32 0xbf1c5f5e Anonymous_GTID last_committed=3 sequence_number=4
- SET @@SESSION.GTID_NEXT= 'ANONYMOUS'/*!*/;
- # at 1097
- #180609 1:11:25 server id 1 end_log_pos 1179 CRC32 0xeebaef40 Query thread_id=10 exec_time=0 error_code=0
- SET TIMESTAMP=1528506685/*!*/;
- BEGIN
- /*!*/;
- # at 1179
- #180609 1:11:25 server id 1 end_log_pos 1235 CRC32 0x6b867c5b Table_map: `dbtest`.`table1` mapped to number 254
- # at 1235
- #180609 1:11:25 server id 1 end_log_pos 1294 CRC32 0xc058c50f Write_rows: table id 254 flags: STMT_END_F
- BINLOG '
- PSkbWxMBAAAAOAAAANMEAAAAAP4AAAAAAAEABmRidGVzdAAGdGFibGUxAAMDDxIDeAAABlt8hms=
- PSkbWx4BAAAAOwAAAA4FAAAAAP4AAAAAAAEAAgAD//gGAAAADWFmdGVyX2JhY2t1cDKZoBIS2Q/F
- WMA=
- '/*!*/;
- # at 1294
- #180609 1:11:25 server id 1 end_log_pos 1325 CRC32 0x8113be5c Xid = 992
- COMMIT/*!*/;
- SET @@SESSION.GTID_NEXT= 'AUTOMATIC' /* added by mysqlbinlog */ /*!*/;
- DELIMITER ;
- # End of log file
- /*!50003 SET COMPLETION_TYPE=@OLD_COMPLETION_TYPE*/;
- /*!50530 SET @@SESSION.PSEUDO_SLAVE_MODE=0*/;
上面清楚的记载了我们id=5和id=6插入的情况,是不是很有意思。
刚才我们看到的是只有1个binlog的情况,而我们知道binlog是会不断增加的,当有两个以上的binlog的时候该怎么办。
沃恩首先到源数据库上,手动建立新的binlog
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | xxxxxxx1 | 2018-06-09 09:43:00 |
- | 5 | xxxxxxx1 | 2018-06-09 09:43:05 |
- +----+---------------+---------------------+
- 5 rows in set (0.00 sec)
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000001
- Position: 2233
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
现在手动创建新的binlog文件
- mysql> flush logs;
- Query OK, 0 rows affected (0.30 sec)
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000002
- Position: 154
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
这样一个新的00002的binlog就开始使用了,起始行数154,并且原来的00001并没有被删掉,而是永久的保存在那里。
我们在创建新binlog后插入几条数据进去
- mysql> insert into table1 values (6,'new_binlog1',NOW());
- Query OK, 1 row affected (0.13 sec)
- mysql> insert into table1 values (7,'new_binlog2',NOW());
- Query OK, 1 row affected (0.15 sec)
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | xxxxxxx1 | 2018-06-09 09:43:00 |
- | 5 | xxxxxxx1 | 2018-06-09 09:43:05 |
- | 6 | new_binlog1 | 2018-06-09 10:47:21 |
- | 7 | new_binlog2 | 2018-06-09 10:47:28 |
- +----+---------------+---------------------+
- 7 rows in set (0.00 sec)
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000002
- Position: 736
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
现在假设系统崩溃,我们要异地恢复这个数据库,我们会有之前的一个全备,和两个binlog
- mysql> show databases;
- +--------------------+
- | Database |
- +--------------------+
- | information_schema |
- | mydb |
- | mysql |
- | performance_schema |
- | sys |
- +--------------------+
- 5 rows in set (0.00 sec)
还是首先恢复全库备份,显而易见,恢复之后只有id=1-4的数据,并且id=4的数据还是旧版本的
- mysql> use dbtest;
- Reading table information for completion of table and column names
- You can turn off this feature to get a quicker startup with -A
- Database changed
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 08:44:33 |
- | 4 | after_binlog2 | 2018-06-09 08:44:38 |
- +----+---------------+---------------------+
- 4 rows in set (0.00 sec)
然后我们需要按照binlog的顺序依次恢复,也就是先重做00001上的,再重做00002上的,由于全备是从00001的740行开始的,所以再重做00001的时候要采用--start-position,而重做00002的时候就不需要了。
# mysqlbinlog --no-defaults --start-position=740 /import/mysql/mysql-bin.000001 | mysql -uroot -p
此时会恢复到我们flush log之前的状态
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 00:44:33 |
- | 4 | xxxxxxx1 | 2018-06-09 01:43:00 |
- | 5 | xxxxxxx1 | 2018-06-09 01:43:05 |
- +----+---------------+---------------------+
- 5 rows in set (0.00 sec)
然后再重做00002上的事务,此时不需要指定--start-postion了,因为00001和00002是连续的。
- # ls
- error.log mysql-bin.000001 mysql-bin.000002 mysql-bin.index
- # mysqlbinlog --no-defaults /import/mysql/mysql-bin.000002 | mysql -uroot -p
发现我们已经把全部数据都恢复回来了
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 00:44:33 |
- | 4 | xxxxxxx1 | 2018-06-09 01:43:00 |
- | 5 | xxxxxxx1 | 2018-06-09 01:43:05 |
- | 6 | new_binlog1 | 2018-06-09 10:47:21 |
- | 7 | new_binlog2 | 2018-06-09 10:47:28 |
- +----+---------------+---------------------+
- 7 rows in set (0.00 sec)
同样,你也可以在恢复00001的时候指定--stop-datetime,但是这样在重做00002的时候可能会碰到主键冲突的问题,需要你自己去把握了。
上面都是DML操作,通过binlog可以准确无误的还原回来,下面再执行一个DDL操作,看看binlog是否还有用
首先我们建立一张新表table2,然后drop掉table1,之后看看能否异地恢复。然后我们再考虑drop table1是一个误操作的情况下,能否异地恢复。
- mysql> create table table2 (
- -> id int primary key,
- -> hobby varchar(40),
- -> starttime datetime)
- -> ;
- Query OK, 0 rows affected (0.55 sec)
- mysql> insert into table2 values(1,'play',NOW());
- Query OK, 1 row affected (0.12 sec)
- mysql> select * from table2;
- +----+-------+---------------------+
- | id | hobby | starttime |
- +----+-------+---------------------+
- | 1 | play | 2018-06-09 11:06:07 |
- +----+-------+---------------------+
- 1 row in set (0.00 sec)
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000002
- Position: 1243
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
可以看到00002已经到了1243行了,下面我们drop table1
- mysql> drop table table1;
- Query OK, 0 rows affected (0.42 sec)
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000002
- Position: 1431
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
可以看到,drop操作也被写入到了binlog。下面我们开始异地恢复
- mysql> show databases;
- +--------------------+
- | Database |
- +--------------------+
- | information_schema |
- | mydb |
- | mysql |
- | performance_schema |
- | sys |
- +--------------------+
- 5 rows in set (0.00 sec)
我们先进行全备份的恢复,然后恢复00001,然后恢复00002,其中先恢复到create table2并插入数据完成的阶段,再恢复到drop table1之后的阶段,全面检查DML的可恢复性
- # mysql -uroot -p < /import/full.sql
- Enter password:
- # mysql -uroot -p
- Enter password:
- mysql> use dbtest;
- Reading table information for completion of table and column names
- You can turn off this feature to get a quicker startup with -A
- Database changed
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 00:44:33 |
- | 4 | after_binlog2 | 2018-06-09 00:44:38 |
- +----+---------------+---------------------+
- 4 rows in set (0.00 sec)
- mysql> select * from table2;
- ERROR 1146 (42S02): Table 'dbtest.table2' doesn't exist
然后恢复00001的全部内容和00002中drop table1之前的内容,由于表2的第一条数据是11:06:07插入成功的,所以我们要恢复到11:06:08才能看到着一条数据,恢复到11:06:07是看不到的。
- # mysqlbinlog --no-defaults --start-position=740 /import/mysql/mysql-bin.000001 | mysql -uroot -p
- Enter password:
- # mysqlbinlog --no-defaults --stop-datetime='2018-06-09 03:06:08' /import/mysql/mysql-bin.000002 | mysql -uroot -p
- Enter password:
- root@5d41cbeb4b4d:/import/mysql# mysql -uroot -p
- Enter password:
- mysql> use dbtest;
- Reading table information for completion of table and column names
- You can turn off this feature to get a quicker startup with -A
- Database changed
- mysql> select * from table1;
- +----+---------------+---------------------+
- | id | name | birthday |
- +----+---------------+---------------------+
- | 1 | befor_binlog1 | 2018-06-09 08:36:33 |
- | 2 | befor_binlog2 | 2018-06-09 08:36:40 |
- | 3 | after_binlog1 | 2018-06-09 00:44:33 |
- | 4 | xxxxxxx1 | 2018-06-09 01:43:00 |
- | 5 | xxxxxxx1 | 2018-06-09 01:43:05 |
- | 6 | new_binlog1 | 2018-06-09 10:47:21 |
- | 7 | new_binlog2 | 2018-06-09 10:47:28 |
- +----+---------------+---------------------+
- 7 rows in set (0.00 sec)
- mysql> select * from table2;
- +----+-------+---------------------+
- | id | hobby | starttime |
- +----+-------+---------------------+
- | 1 | play | 2018-06-09 11:06:07 |
- +----+-------+---------------------+
- 1 row in set (0.00 sec)
此时table1和table2的数据都已经异地恢复成功,可见即使源数据库执行了删表操作也是不妨碍我们恢复的。下面我们再恢复到和源数据库一模一样的状态,就是把上一句的--stop-datetime换成--start-datetime。
# mysqlbinlog --no-defaults --start-datetime='2018-06-09 03:06:08' /import/mysql/mysql-bin.000002 | mysql -uroot -p
- mysql> select * from table1;
- ERROR 1146 (42S02): Table 'dbtest.table1' doesn't exist
- mysql> select * from table2;
- +----+-------+---------------------+
- | id | hobby | starttime |
- +----+-------+---------------------+
- | 1 | play | 2018-06-09 11:06:07 |
- +----+-------+---------------------+
- 1 row in set (0.00 sec)
至此,DDL语句也恢复完成了。
如何简化日常备份
还有一种特殊的备份方式,就是再mysqldump语句中加入--flush-logs这样的话会把当前没有写满的binlog停止,另起一个新的binlog来写,这样就不用在重做binlog的时候添加--start-position语句了,如下
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000002
- Position: 1431
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
- mysql> quit
- Bye
- root@f4d417a2e6ea:/# mysqldump --single-transaction --master-data=2 --triggers --routines --flush-logs --flush-privileges --databases dbtest -p > /backup/full2.sql
- Enter password:
在备份的时候还加入了--flush-privileges,这个是在恢复的时候能够自动赋予相关用户相关权限,如果不加这个更适合主从复制迁移数据 --databases dbtest 是只备份 dbtest这个库,减少备份提及,但是mysql里面的user就会不一致了,这样即使加了--flush-privileges,用户权限还是会丢失,所以这样适合单机恢复,就是在出问题后将当前mysql的库drop掉,然后再执行导入。
在备份完毕之后,产生了一个新的binlog 00003,并从154行开始
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000003
- Position: 154
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.01 sec)
此时我们查看我们备份出来的这个文件,看他的起始文件和位置
- # head full2.sql -n 50
- ......
- --
- -- Position to start replication or point-in-time recovery from
- --
- -- CHANGE MASTER TO MASTER_LOG_FILE='mysql-bin.000003', MASTER_LOG_POS=154;
能看到和备份完之后的master状态是一样的。
然后我们在源库里添加几行数据
- mysql> use dbtest;
- Reading table information for completion of table and column names
- You can turn off this feature to get a quicker startup with -A
- Database changed
- mysql> insert into table2 values(2,'hahaha',NOW());
- Query OK, 1 row affected (0.14 sec)
- mysql> insert into table2 values(3,'enenen',NOW());
- Query OK, 1 row affected (0.14 sec)
- mysql> select * from table2;
- +----+--------+---------------------+
- | id | hobby | starttime |
- +----+--------+---------------------+
- | 1 | play | 2018-06-09 11:06:07 |
- | 2 | hahaha | 2018-06-09 04:09:17 |
- | 3 | enenen | 2018-06-09 04:09:34 |
- +----+--------+---------------------+
- 3 rows in set (0.00 sec)
- mysql> show master status \G
- *************************** 1. row ***************************
- File: mysql-bin.000003
- Position: 726
- Binlog_Do_DB:
- Binlog_Ignore_DB:
- Executed_Gtid_Set:
- 1 row in set (0.00 sec)
可以看到00003到了726行了。现在我们假设源库崩溃,然后我们把full2.sql和binlog 00003都拷贝到另一台服务器上。
开始恢复全备
# mysql -uroot -p < /import/full2.sql
- mysql> select * from table2;
- +----+-------+---------------------+
- | id | hobby | starttime |
- +----+-------+---------------------+
- | 1 | play | 2018-06-09 11:06:07 |
- +----+-------+---------------------+
- 1 row in set (0.00 sec)
不出所料只有一行数据,现在我们要重做00003,由于之前--flush-logs的作用,我们虽然从157行开始,但是无需指定--start-positon了,简化了数据库恢复的过程,而且由于重新启用一个binlog,之前的0001,00002就都可以删掉或者转移走保存起来,节省服务器上的宝贵空间,不然的话我们也许还要等00002写满之后才能转移走。
- # mysqlbinlog --no-defaults /import/mysql/mysql-bin.000003 | mysql -uroot -p
- Enter password:
- mysql> select * from table2;
- +----+--------+---------------------+
- | id | hobby | starttime |
- +----+--------+---------------------+
- | 1 | play | 2018-06-09 11:06:07 |
- | 2 | hahaha | 2018-06-09 04:09:17 |
- | 3 | enenen | 2018-06-09 04:09:34 |
- +----+--------+---------------------+
- 3 rows in set (0.00 sec)
通过这样的参数优化,简化了日常备份的工作量
版权声明:遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。
MySQL回滚到某一时刻数据的方法的更多相关文章
- mysql 复制表结构、表数据的方法
From: http://blog.163.com/yaoyingying681@126/blog/static/109463675201191173221759/ MySQL 添加列,修改列,删除列 ...
- Mysql中查找并删除重复数据的方法
(一)单个字段 1.查找表中多余的重复记录,根据(question_title)字段来判断 代码如下 复制代码 select * from questions where question_title ...
- mysql数据库中导入txt文本数据的方法
安装好MySQL和Navicat 8 for MySQL 通过Navicat 8 for MySQL创建数据库test. 2 在数据库test上创建测试数据表student(主键ID,姓名,年龄,学 ...
- mysql实现随机获取几条数据的方法
版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/xionglangs/article/details/50630758sql语句有几种写法 1:SEL ...
- 【转】mysql实现随机获取几条数据的方法
sql语句有几种写法 1:SELECT * FROM tablename ORDER BY RAND() LIMIT 想要获取的数据条数: 2:SELECT *FROM `table` WHERE i ...
- MySql 按周/月/日统计数据的方法
知识关键词:DATE_FORMAT select DATE_FORMAT(create_time,'%Y%u') weeks,count(caseid) count from tc_case gro ...
- MySQL事务部分回滚-回滚到指定保存点
我们可以在mysql事务处理过程中定义保存点(SAVEPOINT),然后回滚到指定的保存点前的状态. 定义保存点,以及回滚到指定保存点前状态的语法如下. 定义保存点---SAVEPOINT 保存点名; ...
- 利用闪回查看Oracle表历史时刻数据
利用闪回查看Oracle表历史时刻数据 1.查看表历史时刻数据 select * from tab_test AS OF TIMESTAMP to_timestamp('20140917 10:00: ...
- SQL Server 2008 数据库回滚到某个时间点
数据库回滚到时间的的前提: 事务日志完整,数据库在完整恢复模式下进行过一次完整备份,数据库没有进行过还原操作(惨痛教训). 当数据库误操作时,切记冷静,不然问题就是滚雪球, 在不做下一步错误前 可观 ...
随机推荐
- Hexo使用攻略-添加分类及标签
创建"分类"选项 生成"分类"页并添加tpye属性 打开命令行,进入博客所在文件夹.执行命令 hexo new page categories 成功后会提示: ...
- C# WinfForm 控件之dev报表 XtraReport (五) 并排报表
有了前边的基础这个就很简单了,建一个容器报表 在detail,上放两个xrsubReport.再做两个明细报表,分别指定到xrsubreport就可以了
- VUE 中 使用 iview Form组件 enter键防止页面刷新
<Form :label-width="100" inline label-position='left' @keydown.native.enter.prevent =&q ...
- C语言指针和字符串
#include <stdio.h> int main() { /********************************************* * 内存: * 1.常量区 * ...
- linux磁盘分区、挂载、查看
实战: 34 查看本机所有磁盘 fdisk -l 35 查看磁盘挂载情况 lsblk -f 36 39: ...
- C++11的闭包(lambda、function、bind)
c++11开始支持闭包,闭包:与函数A调用函数B相比较,闭包中函数A调用函数B,可以不通过函数A给函数B传递函数参数,而使函数B可以访问函数A的上下文环境才可见(函数A可直接访问到)的变量:比如: 函 ...
- MaxCompute问答整理之9月
本文是基于本人对MaxCompute产品的学习进度,再结合开发者社区里面的一些问题,进而整理成文.希望对大家有所帮助. 问题一.如何查看information_schema的tables? 在使用OD ...
- thinkphp 日志驱动
日志驱动默认的命名空间位于Think\Log\Driver,驱动类需要实现的接口方法包括: 方法 说明 架构方法 __construct($config=array()) 写入方法 write($lo ...
- PHP headers_sent() 函数
定义和用法 headers_sent() 函数检查 HTTP 报头是否发送/已发送到何处. 如果报头已发送,该函数返回 TRUE,否则返回 FALSE. 语法 headers_sent(file,li ...
- SPOJ:[DIVCNT3]Counting Divisors
题目大意:求1~N的每个数因子数的立方和. 题解:由于N过大,我们不能直接通过线性筛求解.我们可以采用洲阁筛. 洲阁筛的式子可以写成: 对于F(1~√n),可以直接线性筛求解. 对于,我们进行以下DP ...