备份时数据库处于关闭状态,直接打包数据库文件
备份速度快,恢复时也是最简单的
专用备份工具mydump或mysqlhotcopy
mysqldump常用的逻辑备份工具
mysqlhotcopy仅拥有备份MyISAM和ARCHIVE表
进行增量备份,需要刷新二进制日志
免费的MySQL热备份软件Percona XtraBackup
mysql数据库备份常见命令
systemctl stop mysqld
mkdir /backup
tar zcvf /backup/mysql_all-$(date +%F).tar.gz /usr/local/mysql/data/
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数据库
使用tar命令直接打包数据库文件夹
直接替换现有MySQL目录即可
mysqldump备份与恢复
MySQL自带的备份工具,可方便实现对MySQL的备份
可以将指定的库、表导出为SQL脚本
使用命令mysql导入备份的数据
mysqldump命令是MySQLI自带的备份工具,相当方便对MySQL进行备份
通过该命令工具可以将指定的库、表或全部的库导出为SQL脚本,在需要恢复时可进行数据恢复
向数据库中添加一些数据
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');
语法:
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.
语法:
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库
语法:
mysqldump -u 用户名 -p [密码] [选项] --all-databases > /备份路径/备份文件名
如:
mysqldump -u root -p --opt --all-databases > /backup/all.sql
在生产环境中,存在对某个特定表的维护操作,此时 mysqldump同样发挥重大作用
语法
mysqldump -u 用户名 -p [密码] [选项] 数据库名 表名 > /备份路径/备份文件名
如:
mysqldump -uroot -pabc123 school cd > /backup/school-cd.sql
mysqldump -u 用户名 -p [密码] [选项] -d 数据库名 表名 > /备份路径/备份文件名
如:
mysqldump -uroot -pabc123 -d school cd > /backup/school-cd_structure.sql
mysqldump命令导出的SQL备份脚本,在进行数据恢复时可使用以下方法导入:
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;
语法:
mysql -u 用户名 -p [密码] < 库备份脚本的路径
如:
mysql -u root -pabc123 < /backup/school.sql
语法
mysql -u 用户名 -p [密码] < 表备份脚本的路径
如:
mysql -u root -p mysql < /backup/school-cd.sql
mysql> source /backup/school-cd.sql;
在生产环境中,可以使用shell脚本自动实现定期备份
只有那些在上次完全备份或者增量备份后被修改的文件才会被备份
增量备份就是备份自上一次备份之后增加或变化的文件或者内容
备份数据中有重复数据
备份时间与恢复时间长
没有重复数据,效率高,空间利用率最大化
备份量不大,时间短
恢复麻烦:需要上次完全备份及完全备份之后所有的增量备份才能恢复,而且要对所有增量备份进行逐个反推恢复
安全性较低
二进制日志保存了所有更新或者可能更新数据库的操作
二进制日志在启动 MySQL服务器后开始记录,并在文件达到max_binlog_size所设置的大小或者接收到 flush-logs命令后重新创建新的日志文件
只需定时执行 flush-logs方法重新创建新的日志,生成二进制文件序列,并及时把这些旧的日志保存到安全的地方就完成了一个时间段的增量备份
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。
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; '查看限制正在使用哪个日志''
语法
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
就是将某个起始时间的二进制日志导入数据库中,从而跳过某个发生错误的时间点实现数据的恢复
mysqlbinlog --stop-position='操作id' 二进制日志 |mysql -u 用户名 -p 密码
mysqlbinlog --start-position='操作id' 二进制日志 |mysql -u 用户名 -p 密码
使用基于时间点的恢复,可能会出现在一个时间点里既同时存在正确的操作又存在错误的操作,所以需要一种更为精确的恢复方式
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 密码
开启二进制日志–>添加数据–>进行完全备份–>录入新的数据–>进行增量备份(刷新二进制日志)–>模拟故障––恢复操作
[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
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
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里面
[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'
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; //看表已经跳过误删除名为王五的操作继续恢复后面插入张六的操作
由于已经基于时间点恢复,所以把数据删除,再用位置点进行恢复
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; //查看表已经跳过误删除名为王五的操作继续恢复后面插入张六的操作