再谈 MySQL 备份
之前已经写过一篇关于 MySQL 备份和恢复 的文章,讲述了几种比较常见的备份和恢复策略。本文则是以一个实际的案例来讲解全量备份和增量备份。
一、前提条件
首先,需要设定一个同步的环境出来,可以在两个独立的主机或者同一个主机上实现,详细请参考MySQL做同步实例。接下来的备份工作就都是在Slave上实现了。
下面就拿一个实际的例子来讲解,MySQL Slave的版本为 5.0.27。
假定 datadir 为 /usr/local/mysql/data,备份文件存放目录为 /backup/mysql/,两种备份都由shell脚本来自动完成。
二、全量备份
全量备份的原理是停止slave之后,将数据刷新到磁盘中,然后直接拷贝数据文件。如果数据量相对较小,推荐使用mysqldump来做,较大的话,直接拷贝文件就更便捷了。
shell脚本文件名 backupmysql_fullly.sh:
#!/bin/sh # # created by yejr,2006/11/29 # # 本脚本用于定期做全量备份,备份的对象是slave上的全部数据 # 每次备份之前都先执行"STOP SLAVE; FLUSH TABLES;",然后将 # 所有文件拷贝到备份目录下 # #取得当前日期,作为备份目录名 today=`date +"%Y_%m_%d"` #源目录 datadir=/usr/local/mysql/data #目标目录 bkdir=/backup/mysql today_bkdir=$bkdir/$today today_full_bkdir=$today_bkdir/full_backup #修改上级目录属性 chattr -i $bkdir #判断目标目录是否已经存在 if [ ! -d $today_bkdir ] ; then mkdir -p $today_bkdir else chattr -i $today_bkdir fi #执行 slave-stop && flush-logs,停止slave,并且刷新缓存内容到磁盘 mysqladmin -hlocalhost -uroot -ppasswd stop-slave mysqladmin -hlocalhost -uroot -ppasswd flush-tables cd $datadir #将ibdata*以及其它需要备份的数据库目录拷贝到目标目录下 cp -rf ib* forum my.cnf mysql news yejr $today_full_bkdir/ #修改文件属主 chown -R nobody:nobody $bkdir #执行 slave-start,启动slave mysqladmin -hlocalhost -uroot -ppasswd start-slave #删除过期文件 cd $bkdir #找出6天前创建的目录 for d in `find . -type d -mtime +6 -maxdepth 1` do #修改当前目录及其子目录属性,然后删除 chattr -i $d chattr -i $d/full_backup rm -fr $d done #增加属性标志位 chattr +i $bkdir
三、增量备份
增量备份的原理是利用binlog的特性,定期刷新binlog,然后将旧的binlog备份起来,再加上以前的全量备份,就能在较短时间内得到当前的运行数据。
shell脚本文件名 backupmysql_hourly.sh:
#!/bin/sh # # created by yejr,2006/11/29 # 本脚本用于定期做增量备份,备份的对象是binlog # 每次备份之前都先执行 flush-logs,然后将最近的binlog移动 # 到备份目录下 # #取得当前日期,作为备份目录名 today=`date +"%Y_%m_%d"` #源目录 datadir=/usr/local/mysql/data #目标目录 bkdir=/backup/mysql today_bkdir=$bkdir/$today #修改上级目录属性 chattr -i $bkdir #判断目标目录是否已经存在 if [ ! -d $today_bkdir ] ; then mkdir -p $today_bkdir else chattr -i $today_bkdir fi #先执行"flush logs",创建出新的log文件 mysqladmin -hlocalhost -uroot -ppasswd flush-logs #去的binlog文件总数,去掉最后一个以及binlog.index total=`ls $logdir/bin_log.*|wc -l` total=`expr $total - 2` for f in `ls $logdir/bin_log.*|head -n $total` do #备份日志文件 bf=`basename $f` mv $f $today_bkdir/$bf done #修改属主及目录属性 chown -R nobody:nobody $bkdir chattr +i $bkdir
最后,在crontab中新增如下两行记录:
#从1-23时,每小时增量备份一次 0 1-23/1 * * * (/bin/sh /backup/mysql/backupmysql_hourly.sh \ >> /backup/mysql/backup.log &) #周一、三、六、日凌晨0点全量备份 0 0 * * 0,1,3,6 (/bin/sh /backup/mysql/backupmysql_fullly.sh \ >> /backup/mysql/backup.log &)
如果觉得备份的频率太低了,可以自己适当调整crontab的频率。
以上脚本来自实际运行环境后稍作修改。
评论
leera (未验证)
周二, 2007/03/20 - 13:41
Permalink
我碰到一个从未遇到
我碰到一个从未遇到的MySQL问题
请看我blog
http://www.leera.cn/read.php?24
能否给出建议?谢谢
hellsp (未验证)
周四, 2007/03/29 - 17:33
Permalink
请问slave本身有没有bin
请问slave本身有没有binlog日志?
我在做试验的过程中,发现slave服务器上,只有relay-binlog里面有master上面执行的语句,而slave本身的binlog中没有master上面执行的sql语句.
是不是slave本身的binlog日志不记录从master上过来的语句?
如果不是,如何配置?
如果是的话,那在slave上刷新增量备份binlog岂不是没有意义了??
carlson
周五, 2007/03/30 - 11:58
Permalink
通常情况,从服务器
通常情况,从服务器从主服务器接收到的更新不记入它的二进制日志。选项--logs-slave-updates告诉从服务器将其SQL线程执行的更新记入到从服务器自己的二进制日志。为了使该选项生效,还必须用--logs-bin选项启动从服务器以启用二进制日志.
银河 (未验证)
周一, 2008/05/05 - 11:39
Permalink
请教这样的方式备份
请教这样的方式备份的话,如何恢复数据呢?因为binlog被移动了,恢复的时候似乎会很麻烦啊……
yejr
周一, 2008/05/05 - 15:58
Permalink
当然是把binlog再拷贝
当然是把binlog再拷贝回来恢复了。也可以在同一个私网里,把binlog释放出来,然后通过私网连接直接恢复。例如:
MySQL方案、培训、支持
MySQL 用户组