MySQL备份与恢复

一: 数据库备份的分类

1.1: 数据备份重要性

  • 在生产环境中,数据的安全性至关重要
  • 任何数据的丢失都可能产生严重的后果
    造成数据丢失的原因
  1. 程序错误
  2. 人为操作失误
  3. 运算错误
  4. 磁盘故障
  5. 灾难(如火灾、地震)和盗窃

1.2 分类

  • 从物理与逻辑的角度,备份可分为
  1. 物理备份:对数据库操作系统的物理文件(如数据文件、日志文件等)的备份
  • 物理备份方法
    1.1 冷备份(脱机备份) :是在关闭数据库的时候进行的,依赖于数据文件
    1.2 热备份(联机备份) :数据库处于运行状态,依赖于数据库的日志文件
    1.3 温备份:数据库锁定表格(不可写入但可读)的状态下进行备份操作
  1. 逻辑备份:对数据库逻辑组件(如: 表等数据库对象)的备份
  • 按照数据库的备份策略角度,备份可分为
  1. 完全备份:每次对数据进行完整的备份
  2. 差异备份:备份那些自从上次完全备份之后被修改过的文件
  3. 增量备份:只有那些在上次完全备份或者增量备份后被修改的文件才会被备份

1.3 常见的备份

  1. 物理冷备

备份时数据库处于关闭状态,直接打包数据库文件
备份速度快,恢复时也是最简单的
专用备份工具mydump或mysqlhotcopy
mysqldump常用的逻辑备份工具
mysqlhotcopy仅拥有备份MyISAM和ARCHIVE表

  1. 启用二进制日志进行增量备份

进行增量备份,需要刷新二进制日志

  1. 第三方工具备份

免费的MySQL热备份软件Percona XtraBackup

      mysql数据库备份常见命令

1.4 MySQL物理冷备份与恢复

  1. 关闭数据库,打包备份
systemctl stop mysqld
mkdir /backup
tar zcvf /backup/mysql_all-$(date +%F).tar.gz /usr/local/mysql/data/
  1. 恢复数据库,将备份数据mv成线上库文件夹
mkdir /bak
mv /usr/local/mysql/data/ /bak/
mkdir restore
tar zxvf /backup/mysql_all-2020-12-24.tar.gz -C restore/
mv restore/usr/local/mysql/data/ /usr/local/mysql/
systemctl start mysqld

二:MySQL的完全备份

2.1:概念

  • 完全备份是对整个数据库的备份、数据库结构和文件结构的备份

  • 完全备份保存的是备份完成时刻的数据库

  • 完全备份是增量备份的基础

2.2 优缺点

优点:

  • 安全性高
  • 备份与恢复操作简单方便

缺点:

  • 数据存在大量的重复
  • 占用大量的备份空间,空间利用率低
  • 备份与恢复时间长

2.3 mysqldump备份库

2.3.1 完全备份MySQL数据库分类

物理冷备份与恢复

关闭MySQL数据库
使用tar命令直接打包数据库文件夹
直接替换现有MySQL目录即可

mysqldump备份与恢复

MySQL自带的备份工具,可方便实现对MySQL的备份
可以将指定的库、表导出为SQL脚本
使用命令mysql导入备份的数据
  • mysqldump介绍

mysqldump命令是MySQLI自带的备份工具,相当方便对MySQL进行备份
通过该命令工具可以将指定的库、表或全部的库导出为SQL脚本,在需要恢复时可进行数据恢复

2.3.2 mysqldump命令备份数据库

向数据库中添加一些数据

mysql> create database school;
mysql> use school;
mysql> create table cd(id int(4) auto_increment primary key, name varchar(16), age char(2));
mysql> insert into cd(name,age) values('zhangsan','18'),('lisi','19');

在这里插入图片描述

  1. 备份单个库

语法:

mysqldump -u 用户名 -p [密码] [选项] [库名] > /备份路径/备份文件名
mysqldump -u root -p cd > /backup/cd.sql

示例

mysqldump -uroot -pabc123 school > /opt/school.sql
mysqldump: [Warning] Using a password on the command line interface can be insecure.

在这里插入图片描述

  1. 备份多个库

语法:

mysqldump -u 用户名 -p [密码] [选项] --databases 库名1 [库名2] ...  > /备份路径/备份文件名

如:

mysqldump -u root -p --databases yiku erku > /backup/yiku-erku.sql

示例:

mysqldump -uroot -pabc123 --databases school name > /opt/school-name.sql     //同时备份school和name库
  1. 对所有库备份

语法:

mysqldump -u 用户名 -p [密码] [选项] --all-databases > /备份路径/备份文件名

如:

mysqldump -u root -p --opt --all-databases > /backup/all.sql

2.3.3 mysqldump命令备份表

在生产环境中,存在对某个特定表的维护操作,此时 mysqldump同样发挥重大作用

  • 使用 mysqldump备份表的操作

语法

mysqldump -u 用户名 -p [密码] [选项] 数据库名 表名 > /备份路径/备份文件名

如:

mysqldump -uroot -pabc123 school cd > /backup/school-cd.sql

在这里插入图片描述

  • 使用 mysqldump备份表的结构
mysqldump -u 用户名 -p [密码] [选项] -d 数据库名 表名 > /备份路径/备份文件名

如:

mysqldump -uroot -pabc123 -d school cd > /backup/school-cd_structure.sql

在这里插入图片描述

三、MySQL数据恢复

mysqldump命令导出的SQL备份脚本,在进行数据恢复时可使用以下方法导入:

  1. source命令
  2. mysql命令

3.1 source命令恢复数据库

  1. 登录到 MySQL数据库
mysql> source /backup/school-cd.sql;     //使用绝对路径
'//source命令在mysql库中使用'
mysql> drop database school;
mysql> source /backup/school-cd.sql;

在这里插入图片描述
注意: 备份文件里没有判断库的语句,所以要先创建库,再导入表,但是备份多个库的时候,备份的sql语句包含了库的判断,所以不用提前创建库’

mysql> create database school;
mysql> use school
mysql> source /backup/school-cd.sql;
mysql> select * from cd;

在这里插入图片描述

3.2 mysql命令恢复数据库

语法:

mysql -u 用户名 -p [密码] < 库备份脚本的路径

如:

mysql -u root -pabc123 < /backup/school.sql

3.3 恢复数据表

  • 恢复表时同样可以使用 source或者mysql命令
  • source恢复表的操作与恢复库的操作相同
  • 当备份文件中只包含表的备份,而不包括创建库的语句时,必须指定库名,且目标库必须存在

语法

  1. mysql命令
mysql -u 用户名 -p [密码] < 表备份脚本的路径

如:

mysql -u root -p mysql < /backup/school-cd.sql
  1. source命令
mysql> source /backup/school-cd.sql;

在生产环境中,可以使用shell脚本自动实现定期备份

四: MySQL增量备份与恢复

4.1 概述

只有那些在上次完全备份或者增量备份后被修改的文件才会被备份
增量备份就是备份自上一次备份之后增加或变化的文件或者内容

  • mysqldump命令完全备份存在的问题

备份数据中有重复数据
备份时间与恢复时间长

4.2 优缺点

  • 优点:

没有重复数据,效率高,空间利用率最大化

备份量不大,时间短

  • 缺点:

恢复麻烦:需要上次完全备份及完全备份之后所有的增量备份才能恢复,而且要对所有增量备份进行逐个反推恢复

安全性较低

4.3 MySQL二进制日志

  • 二进制日志保存了所有更新或者可能更新数据库的操作

  • 二进制日志在启动 MySQL服务器后开始记录,并在文件达到max_binlog_size所设置的大小或者接收到 flush-logs命令后重新创建新的日志文件

  • 只需定时执行 flush-logs方法重新创建新的日志,生成二进制文件序列,并及时把这些旧的日志保存到安全的地方就完成了一个时间段的增量备份

4.4 MySQL 增量备份

MySQL没有提供直接的增量备份方法

  • 可以通过 MySQL提供的二进制日志( binary logs)间接实现增量备份

  • MySQL的配置文件的[mysqld]项中加入log-bin=filepath项(filepath是二进制文件的路径),如log-bin=mysql-bin,然后重启mysqld服务。

  • 二进制日志文件的默认路径为/usr/local/mysql/data

[root@localhost ~]# vim /etc/my.cnf
'//在[mysqld]项中加入配置 log-bin=mysql-bin'
...
server-id = 1
log-bin=mysql-bin	        '//添加此句'
systemctl restart mysqld	'//重启服务'

二进制日志文件的默认路径为/usr/local/mysql/data

  • 语法
    在这里插入图片描述
mysqladmin -u 用户名 -p [密码] flush-logs

只需定时执行 flush-logs方法重新创建新的日志,生成二进制文件序列,并及时把这些旧的日志保存到安全的地方就完成了一个时间段的增量备份。
在这里插入图片描述
可以看出:多了一个文件mysql-bin.000002。

4.5 MySQL 增量备份恢复

4.5.1 查看到日志操作语句

mysqlbinlog --no-defaults --base64-output=decode-rows -v 日志文件名称 /opt/mysql.log.txt	
  '//使用64位解码器按行输出日志文件放到/opt/mysql.log.txt中'
cat /opt/mysql.log.txt  '//查看日志文件的详细信息'

mysql> show master status;                       '查看限制正在使用哪个日志''

在这里插入图片描述

4.5.2 一般恢复

语法

mysqlbinlog [--no-defaults] 增量备份文件 | mysql -u 用户名 -p

基于二进制日志恢复

mysql> use school;
mysql> drop table cd;
mysql> source /backup/school-cd.sql

或者

[root@mysql data]# mysqlbinlog --no-defaults mysql-bin.000001 | mysql -uroot -pabc123

4.5.3 断点恢复

  1. 基于位置恢复

就是将某个起始时间的二进制日志导入数据库中,从而跳过某个发生错误的时间点实现数据的恢复

  • 恢复数据到指定位置
mysqlbinlog --stop-position='操作id' 二进制日志 |mysql -u 用户名 -p 密码
  • 从指定的位置开始恢复数据
mysqlbinlog --start-position='操作id' 二进制日志 |mysql -u 用户名 -p 密码
  1. 基于时间点恢复

使用基于时间点的恢复,可能会出现在一个时间点里既同时存在正确的操作又存在错误的操作,所以需要一种更为精确的恢复方式

  • 从日志开头截止到某个时间点的恢复
mysqlbinlog [--no-defaults] --stop-datetime='年-月-日 小时:分钟:秒' 二进制日志 |mysql -u 用户名 -p 密码
  • 从某个时间点到日志结尾的恢复
mysqlbinlog [--no-defaults] --start-datetime='年-月-日 小时:分钟:秒' 二进制日志 |mysql -u 用户名 -p 密码
  • 从某个时间点到某个时间点的恢复
mysqlbinlog [--no-defaults] --start-datetime='年-月-日 小时:分钟:秒' --stop-datetime='年-月-日 小时:分钟:秒' 二进制日志 |mysql -u 用户名 -p 密码

4.6 增量恢复步骤

开启二进制日志–>添加数据–>进行完全备份–>录入新的数据–>进行增量备份(刷新二进制日志)–>模拟故障––恢复操作

  1. 开启二进制日志
[root@mysql ~]# vim /etc/my.cnf
...
server-id = 1
log-bin=mysql-bin                 '//添加此行开启二进制日志,名称为mysql-bin开头'
[root@mysql ~]# cd /usr/local/mysql/data
[root@mysql data]# ls
apple           ibdata1      ibtmp1  performance_schema
auto.cnf        ib_logfile0  kgc     school
ib_buffer_pool  ib_logfile1  mysql   sys
[root@mysql data]# systemctl restart mysqld       //开启二进制日志后重启服务
[root@mysql data]# ls
apple           ibdata1      ibtmp1  mysql-bin.000001    school
auto.cnf        ib_logfile0  kgc     mysql-bin.index     sys
ib_buffer_pool  ib_logfile1  mysql   performance_schema
                                      //生成二进制日志文件mysql-bin.000001
  1. 进行完全备份
mysql> use school;
mysql> select * from cd;
+----+----------+------+
| id | name     | age  |
+----+----------+------+
|  1 | zhangsan | 18   |
|  2 | lisi     | 19   |
+----+----------+------+
2 rows in set (0.00 sec)
[root@mysql ~]# mysqldump -uroot -pabc123 school cd > /opt/school-cd2.sql

在这里插入图片描述

  1. 进行正常操作和误操作,进行增量备份
mysql> use school;
mysql> insert into cd values(3,'wangwu',20);     //正常操作

mysql> delete from cd where name='zhangsan';           //误操作

mysql> insert into cd values(5,'zhangliu',21);     //正常操作
mysql> select * from cd;

在这里插入图片描述

[root@mysql data]# mysqladmin -uroot -pabc123 flush-logs   //刷新日志增量备份
[root@mysql data]# ls -lh                                   
-rw-r-----. 1 mysql mysql  201 12 24 18:51 mysql-bin.000001
-rw-r-----. 1 mysql mysql 1.0K 12 24 19:42 mysql-bin.000002
-rw-r-----. 1 mysql mysql  154 12 24 19:42 mysql-bin.000003

 //刷新日志后生成000002文件,原来的正常操作和误操作保存在日志000001里面
  1. 查看增量备份二进制文件
[root@mysql data]# mysqlbinlog --no-defaults --base64-output=decode-rows -v mysql-bin.000001 > /opt/bak.txt
[root@mysql data]# vim /opt/bak.txt
......
BEGIN
/*!*/;
# at 293
#201224 19:38:38 server id 1  end_log_pos 346 CRC32 0xf7359359  Table_map: `school`.`cd` mapped to number 222
# at 346
#201224 19:38:38 server id 1  end_log_pos 396 CRC32 0xc34a1362  Write_rows: table id 222 flags: STMT_END_F
### INSERT INTO `school`.`cd`
### SET
###   @1=3
###   @2='wangwu'
###   @3='20'
# at 396
#201224 19:38:38 server id 1  end_log_pos 427 CRC32 0xa5b658aa  Xid = 173
COMMIT/*!*/;
# at 427
#201224 19:39:32 server id 1  end_log_pos 492 CRC32 0x812226d5  Anonymous_GTID  last_committed=1        sequence_number=2       rbr_only=yes
/*!50718 SET TRANSACTION ISOLATION LEVEL READ COMMITTED*//*!*/;
SET @@SESSION.GTID_NEXT= 'ANONYMOUS'/*!*/;
# at 492
#201224 19:39:32 server id 1  end_log_pos 566 CRC32 0x61231674  Query   thread_id=17    exec_time=0     error_code=0
SET TIMESTAMP=1608809972/*!*/;
BEGIN
/*!*/;
# at 566
#201224 19:39:32 server id 1  end_log_pos 619 CRC32 0x7a6e973e  Table_map: `school`.`cd` mapped to number 222
# at 619
#201224 19:39:32 server id 1  end_log_pos 671 CRC32 0x24435a14  Delete_rows: table id 222 flags: STMT_END_F
### DELETE FROM `school`.`cd`
### WHERE
###   @1=1
###   @2='zhangsan'
###   @3='18'
# at 671
#201224 19:39:32 server id 1  end_log_pos 702 CRC32 0xc9db2a9b  Xid = 174
COMMIT/*!*/;
# at 702
#201224 19:40:19 server id 1  end_log_pos 767 CRC32 0x4ae2a49c  Anonymous_GTID  last_committed=2        sequence_number=3       rbr_only=yes
/*!50718 SET TRANSACTION ISOLATION LEVEL READ COMMITTED*//*!*/;
SET @@SESSION.GTID_NEXT= 'ANONYMOUS'/*!*/;
# at 767
#201224 19:40:19 server id 1  end_log_pos 841 CRC32 0xa41b163a  Query   thread_id=17    exec_time=0     error_code=0
SET TIMESTAMP=1608810019/*!*/;
BEGIN
/*!*/;
# at 841
#201224 19:40:19 server id 1  end_log_pos 894 CRC32 0x2c22cb13  Table_map: `school`.`cd` mapped to number 222
# at 894
#201224 19:40:19 server id 1  end_log_pos 946 CRC32 0xfe387271  Write_rows: table id 222 flags: STMT_END_F
### INSERT INTO `school`.`cd`
### SET
###   @1=5
###   @2='zhangliu'
###   @3='21'
# at 946
#201224 19:40:19 server id 1  end_log_pos 977 CRC32 0x40cd330a  Xid = 175
COMMIT/*!*/;
# at 977
#201224 19:42:30 server id 1  end_log_pos 1024 CRC32 0x7096e131         Rotate to mysql-bin.000003  pos: 4
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*/;
......
'注意时间201224 19:38:38 对应的是 2020-12-24 19:38:38'
'时间200821 19:40:19  对应的格式是 2020-12-24 19:40:19'
  1. 使用基于时间点的断点恢复
mysqlbinlog --no-defaults --stop-datetime='2020-12-24 19:38:38' /usr/local/mysql/data/mysql-bin.000001 | mysql -u root -pabc123
'//第一个正常操作的结尾时间是'2020-12-24 19:38:38',所以此处用此时间,表示恢复到此时间的操作'
mysqlbinlog --no-defaults --start-datetime='2020-12-24 19:40:19' /usr/local/mysql/data/mysql-bin.000002 | mysql -u root -pabc123
'//跳过中间失误操作失误操作后的正常操作的开始时间是'2020-12-24 19:40:19',所以此处用此时间,表示从此时间恢复到结尾的操作'
mysql> select * from kgc;          //看表已经跳过误删除名为王五的操作继续恢复后面插入张六的操作

在这里插入图片描述

  1. 使用基于位置点的断点恢复

由于已经基于时间点恢复,所以把数据删除,再用位置点进行恢复

mysql> drop table cd;                            //删除表
mysql> source /opt/school-cd.sql;                       
mysqlbinlog --no-defaults --stop-position='612' /usr/local/mysql/data/mysql-bin.000001 | mysql -u root -pabc123
mysql: [Warning] Using a password on the command line interface can be insecure. 
//这里612对应误操作664的上一个位置点,这里千万不要写664,写了664这个位置,就说明把这个位置的语句执行一遍;
恢复的时候没有成功跳过断点,记住断点的上一个位置点
mysql> select * from school.cd;

mysqlbinlog --no-defaults --start-position='806' /usr/local/mysql/data/mysql-bin.000001 | mysql -u root -pabc123
mysql: [Warning] Using a password on the command line interface can be insecure.
//806位置点为误操作的下一个位置点
mysql> select * from school.cd;        //查看表已经跳过误删除名为王五的操作继续恢复后面插入张六的操作


Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐