实验环境 centos6.4 x86_64 ,mysql编译安装 版本5.5.37

系统关闭 防火墙 seLinux

一 测试环境准备

1.1 编辑/etc/my.cnf 把二进制日志存放到其他非数据目录,innodb每表一个表空间

建立存放 二进制日志 目录

mkdir /binlog

chown mysql:mysql /binlog

修改my.cnf

vim /etc/my.cnf

log-bin = /binlog/mysql-bin #配置文件有此选项直接修改即可,不用添加

innodb_file_per_table = 1

重启mysqld

service mysqld restart

1.2 创建测试数据库与测试表

mysql> create database db;

mysql> use db;

mysql> create table db1(id int primary key auto_increment,name varchar(10),phone int,sex char(10));

mysql> insert into db1 (name,phone,sex) values ('tom',1123213,'m');

mysql> insert into db1 (name,phone,sex) values ('jar',1123,'m');

mysql> insert into db1 (name,phone,sex) values ('xiaozhang',1127,'m');

1.3 创建用于存放备份目录

mkdir /backup

chown –R mysql:mysql /backup

二 用mysqldump备份

案例1 备份单一数据库,删除后,恢复

1 备份数据库

mysqldump -uroot -predhat --master-data=2 --databases db > /backup/db.sql

2 删除数据库

mysql> drop database db;

3 恢复数据库

mysql> source /backup/db.sql # 如恢复不成功,可先建立数据库,再恢复

案例2 完整备份+增量备份+二进制日志

1 完整热备份

mysqldump -uroot -predhat --single-transaction --events --routines --triggers --master-data=2 --databases db >/backup/db_`date +%F`.sql

2 假设第二天 做一些相关操作,做增量备份

mysql> create table db2(id int);

mysql> insert into db2 values (1),(2);

查看当前日志文件位置

mysql> show master status;    
+--------------+----------+--------------+------------------+    
| File | Position | Binlog_Do_DB | Binlog_Ignore_DB |    
+--------------+----------+--------------+------------------+    
| mysql.000010 | 13452 | | |    
+--------------+----------+--------------+------------------+

查看上次完整备份 日志文件位置

less db_2014-06-20.sql

CHANGE MASTER TO MASTER_LOG_FILE='mysql.000010', MASTER_LOG_POS=12977;

做增量备份

mysqlbinlog --start-position=12977 --stop-position=13452 mysql.000010 > /backup/db_`date +%F_%H`.sql

3假设第三天 出现故障 恢复数据库

mysql> insert into db2 values (3),(4);

mysql> insert into db2 values (5),(6),(7),(8);

mysql> drop database db;

恢复数据

mysql>show master status;

查看当前日志位置  如  mysql.000017

mysqlbinlog mysql.000017  

 找到最后操作正确的位置

mysqlbinlog --start-position=13452 --stop-position=13828 mysql.000010 > /tmp/db.sql

mysql> set sql_log_bin=0; #关闭二进制日志

mysql> flush logs;

mysql -uroot -predhat < /backup/db_2014-06-20.sql    
mysql -uroot -predhat < /backup/db_2014-06-20_18.sql    
mysql -uroot -predhat < /tmp/db.sql

mysql> flush privileges; 重读授权文件

mysql> set sql_log_bin=1

 

三 LVM 快照 备份 恢复

( 事务日志和快照必须在一个逻辑卷上)

( sync_binlog =1 防止事务丢失,事务提交,立即同步到磁盘 一般不设为1 备份完成 重新改为 0 )

1 迁移数据库至LVM

备份数据库

mysqldump -uroot -predhat --lock-all-tables --all-databases --routines --triggers  --events > /backup/db_all_`date +%F`.sql

停止mysqld

service mysqld stop

删除原数据库

rm –rf /mydata/*

创建LVM 过程略

建立LVM卷,挂载到/mydata/data

初始化mysql的数据目录 /mydata/data

Vim /etc/my.cnf

sync_binlog=1

恢复数据

mysql> set sql_log_bin=0;

mysql> source /backup/db_all_2014-06-20.sql

mysql> flush priviledges; # 密码生效

mysql> set sql_log_bin=1;

2 快照 备份

mysql> flush tables with read lock; #把所有表从内存刷新到磁盘 ,并请求全局锁

mysql> flush logsl;

mysql> show master status;    
+--------------+----------+--------------+------------------+    
| File | Position | Binlog_Do_DB | Binlog_Ignore_DB |    
+--------------+----------+--------------+------------------+    
| mysql.000017 | 107 | | |    
+--------------+----------+--------------+------------------+    
1 row in set (0.00 sec)    

不要退出此会话

另开一个会话

保存当前日志位置

mysql -uroot -predhat -e 'show master status' > /backup/binlog.txt  

创建快照

lvcreate -L 100M -s -p r -n mydata-snap /dev/myvg/mydata

在前一个会话中 释放读锁

mysql > unlock tables;

挂载快照,拷贝出来,卸载快照,删除快照

mount /dev/myvg/mydata /mnt

mkdir /backup/lvm

cp -pR /mnt/* /backup/lvm/

umont /mnt

lvremove /dev/myvg/mydata-snap

测试数据

service mysqld stop

rm -rf /mydata/data/*

cp -Rp /backup/lvm/data/* /mydata/data/

service mysqld start

恢复后 数据如不完整 ,可使用二进制日志恢复

mysqlbinlog --start-position=107 mysql.000017 > /backup/snap.sql;

mysql> set sql_log_bin=0;

mysql> source /backup/snap.sql

mysql> set sql_log_bin=1;

 

四 xtrabackup

1 下载安装 xtrabackup

yum install libaio

yum install perl-dbd-mysql

yum install perl-Time-HiRes

rpm –ivh percona-xtrabackup-2.1.4-656.rhel6.x86_64.rpm

2 为备份建立一个只有备份权限的用户

mysql> create user 'user'@'localhost' identified by '1234567';

mysql> revoke all privileges ,grant option from 'user'@'localhost';

mysql> grant reload,lock tables,replication client on *.* to 'user'@'localhost';

mysql> flush privileges;

案例1 完整备份mysql

1 完整备份

innobackupex -uroot -predhat /backup/

数据会完整备份到/backup/ 以当期日期时间为目录名的目录下面

(1)xtrabackup_checkpoints —— 备份类型(如完全或增量)、备份状态(如是否已经为prepared状态)和LSN(日志序列号)范围信息;

每个InnoDB页(通常为16k大小)都会包含一个日志序列号,即LSN。LSN是整个数据库系统的系统版本号,每个页面相关的LSN能够表明此页面最近是如何发生改变的。

(2)xtrabackup_binlog_info —— mysql服务器当前正在使用的二进制日志文件及至备份这一刻为止二进制日志事件的位置。

(3)xtrabackup_binary —— 备份中用到的xtrabackup的可执行文件;

(4)backup-my.cnf —— 备份命令用到的配置选项信息;

2 停止mysql服务,删除数据库

service mysqld stop

rm –rf /mydata/data

3 准备恢复阶段

innobackupex --apply-log /backup/2014-06-21_09-35-21/

--apply-log 把未提交的事务回滚,已提交的事务同步到磁盘

4 恢复数据库

innobackupex --copy-back /backup/2014-06-21_09-35-21/

5 修改数据库目录

chown –R mysql:mysql /mydata/data

6 重启服务

service mysqld start

7 再次做完全备份 每次恢复数据后要重新做一个完全备份

innobackupex -uroot -predhat /backup/

案例2 完整备份 +增量备份+二进制日志

假设第一天 做完整备份

innobackupex -uroot -predhat /backup/

第二天 做增量备份

innobackupex --user=root --password=redhat --incremental /backup/ --incremental-basedir=/backup/2014-06-21_10-40-36/

--incremental-basedir 指定基于那个完整备份做增量

第三天 数据库崩溃

数据库服务停止,删空数据库目录

准备恢复数据

innobackupex --apply-log --redo-only /backup/2014-06-21_10-40-36/

--redo-only 未提交的事务不回滚,这个事务可能在增量备份中提交

innobackupex --apply-log --redo-only /backup/2014-06-21_10-40-36/ --incremental-dir=/backup/2014-06-21_10-48-58/

把增量备份添加到完整备份

恢复数据

innobackupex --copy-back /backup/2014-06-21_10-40-36

chown -R mysql:mysql /mydata/data

service mysql start

此时数据应该恢复到第二天 ,如要恢复第三天的数据 可利用二进制日志

进入完整备份目录

cat xtrabackup_binlog_info

mysql.000020 288

查看上次增量备份 二进制日志文件 位置

mysqlbinlog --start-position=288 mysql.000020 > /backup/db.sql

mysql> set sql_log_bin=0;

mysql> source /backup/db.sql

mysql> set sql_log_bin=1;