MySQL 备份和恢复策略

本文详细介绍四种MySQL备份策略:直接拷贝数据库文件、使用mysqlhotcopy、使用mysqldump及主从复制机制。针对不同规模数据库,提供具体脚本与实施步骤。
转自:http://blog.sina.com.cn/s/blog_4e424e2101000c1x.html

在数据库表丢失或损坏的情况下,备份你的数据库是很重要的。如果发生系统崩溃,你肯定想能够将你的表尽可能丢失最少的数据恢复到崩溃发生时的状态。本文主要对MyISAM表做备份恢复。

备份策略一:直接拷贝数据库文件(不推荐)

备份策略二: 使用mysqlhotcopy备份数据库(完全备份,适合小型数据库备份)

备份策略三:使用 mysqldump 备份数据库(完全+增量备份,适合中型数据库备份)

备份策略四:使用 主从复制机制(replication) (实现数据库实时备份)

脚本下载地址: 点击下载脚本

备份策略一、直接拷贝数据库文件

直接拷贝 数据文件最为直接、快速、方便, 缺点是基本上不能实现增量备份。为了保证数据的一致性,需要在 备份 文件前,执行以下SQL语句:

FLUSHTABLESWITHREADLOCK;

也就是把内存中的数据都刷新到磁盘中,同时锁定数据表,以保证拷贝过程中不会有新的数据写入。这种方法备份出来的数据恢复也很简单,直接拷贝回原来的数据库目录下即可。

备份策略二、 使用mysqlhotcopy备份数据库

mysqlhotcopy是一个PERL程序,最初由TimBunce编写。它使用LOCKTABLES、FLUSHTABLES和cp或scp来快速备份数据库。它是备份数据库或单个表的最快的途径,但它只能运行在数据库文件(包括数据表定义文件、数据文件、索引文件)所在的机器上 ,并且 mysqlhotcopy只能用于备份MyISAM 表。

本备份策略适合于小型数据库的备份,数据量不大,可以采用 mysqlhotcopy程序每天进行一次完全备份。

备份策略布置:

(1)、安装DBD-mysqlperl模块,支持mysqlhotcopy脚本连接到MySQL数据库。

shell>tar-xzvfDBD-mysql-4.005.tar.gz

shell>cdDBD-mysql-4.005

shell>unsetLANG

shell>perlMakefile.PL-mysql_config=/usr/local/mysql/bin/mysql_config-testuser=root-testpassword=UserPWD

shell>make

shell>maketest

shell>makeinstall

(2) 、设置crontab任务,每天执行备份脚本

shell>crontab-e

03***/root/MySQLBackup/mysqlbackup.sh>/dev/null2>&1

每天凌晨3:00执行备份脚本。

mysqlbackup.sh注释:

#!/bin/sh

# Name:mysqlbackup.sh

# PS:MySQL DataBase Backup,Use mysqlhotcopy script.

# Write by:i.Stone

# Last Modify:2007-11-15

#

# 定义变量,请根据具体情况修改

# 定义脚本所在目录

scriptsDir=`pwd`

# 数据库的数据目录

dataDir=/usr/local/mysql/data/

# 数据备份目录

tmpBackupDir=/tmp/tmpbackup/

backupDir=/tmp/mysqlbackup/

# 用来备份数据库的用户名和密码

mysqlUser=root

mysqlPWD=111111

# 定义eMail地址

eMail=alter@somode.com


# 如果临时备份目录存在,清空它,如果不存在则创建它

if [[ -e $tmpBackupDir ]]; then

rm -rf $tmpBackupDir/*

else

mkdir $tmpBackupDir

fi

# 如果备份目录不存在则创建它

if [[ ! -e $backupDir ]];then

mkdir $backupDir

fi


# 清空MySQLBackup.log

if [[ -s MySQLBackup.log ]]; then

cat /dev/null >MySQLBackup.log

fi


# 得到数据库备份列表,在此可以过滤不想备份的数据库

for databases in `find $dataDir -type d | /

sed -e "s///usr//local//mysql//data//// " | /

sed -e "s/test//"`; do


if [[ $databases == "" ]]; then

continue

else

# 备份数据库

/usr/local/mysql/bin/mysqlhotcopy --user=$mysqlUser --password=$mysqlPWD -q "$databases" $tmpBackupDir

dateTime=`date "+%Y.%m.%d %H:%M:%S"`

echo "$dateTime Database:$databases backup success!" >>MySQLBackup.log

fi

done


# 压缩备份文件

date=`date -I`

cd $tmpBackupDir

tar czf $backupDir/mysql-$date.tar.gz ./


# 发送邮件通知

if [[ -s MySQLBackup.log ]]; then

cat MySQLBackup.log | mail -s "MySQL Backup" $eMail

fi


# 使用smbclientmv.sh脚本上传数据库备份到备份服务器

# $scriptsDir/smbclientmv.sh

smbclientmv.sh注释

#!/bin/sh

# Name:smbclientmv.sh

# PS:Move the data to Backup Server.

# Write by:i.Stone

# Last Modify:2007-11-15

#

# 定义变量

# 备份服务器名

BackupServer="BackupServerName "

# 共享文件夹名

BackupShare="ShareName "

# 备份服务器的访问用户名和密码

BackupUser="SMBUser "

BackupPW="SMBPassword "

# 定义备份目录

BackupDir=/tmp/mysqlbackup

date=`date -I`


# Move the data to BackupServer

smbclient //$BackupServer/$BackupShare /

$BackupPW -d0 -W WORKGROUP -U $BackupUser /

-c "put $BackupDir/mysql-$date.tar.gz /

mysql-$date.tar.gz"


# Delete temp files

rm -f $BackupDir/mysql-$date.tar.gz

(3)、恢复数据库到备份时的状态
mysqlhotcopy 备份出来的是整个数据库目录,使用时可以直接拷贝到 mysqld 指定的 datadir (在这里是 /usr/local/mysql/data/)目录下即可,同时要注意权限的问题,如下例:
shell> cp -rf db_name /usr/local/mysql/data/
shell> chown -R mysql:mysql /usr/local/mysql/data/ (将 db_name 目录的属主改成 mysqld 运行用户)
本套备份策略只能恢复数据库到最后一次备份时的状态,要想在崩溃时丢失的数据尽量少应该更频繁的进行备份,要想恢复数据到崩溃时的状态请使用主从复制机制(replication)。

备份策略三、 使用 mysqldump 备份数据库

mysqldump是采用SQL级别的备份机制,它将数据表导成SQL脚本文件,在不同的MySQL版本之间升级时相对比较合适,这也是最常用的备份方法。 mysqldump比直接拷贝要慢些。关于mysqldump的更详细解释见最后的附录。

对于中等级别业务量的系统来说,备份策略可以这么定:第一次 完全 备份,每天一次增量备份,每周再做一次 完全 备份,如此一直重复。而对于重要的且繁忙的系统来说,则可能需要每天一次全量备份,每小时一次增量备份,甚至更频繁。为了不影响线上业务,实现在线备份,并且能增量备份,最好的办法就是采用主从复制机制(replication),在slave机器上做备份。

备份策略布置:

(1) 、创建备份目录

Shell>mkdir/tmp/mysqlbackup

Shell>mkdir/tmp/mysqlbackup/daily

(2) 、启用二进制日志

采用binlog的方法相对来说更灵活,省心省力,而且还可以支持增量备份。

启用binlog时必须要重启mysqld。首先,关闭mysqld,打开 /etc/ my.cnf,加入以下几行:

[mysqld]

log-bin

然后启动mysqld就可以了。运行过程中会产生 HOSTNAME-bin .000001以及 HOSTNAME-bin .index,前面的文件是mysqld记录所有对数据的更新操作,后面的文件则是所有binlog的索引,都不能轻易删除。关于binlog 的更详细 信息请查看手册。

(3)、配置SSH密钥登录,用于将MySQL备份传送到备份服务器(如果备份服务器为Windows,请跳过此部)。

1)、在MySQL所在服务器(192.168.0.20)生成SSH密钥

[root@lab~]# ssh-keygen-trsa

Generatingpublic/privatersakeypair.

Enterfileinwhichtosavethekey(/root/.ssh/id_rsa): //直接回车

Enterpassphrase(emptyfornopassphrase): //直接回车,不使用密码

Entersamepassphraseagain: //直接回车,不使用密码

Youridentificationhasbeensavedin/root/.ssh/id_rsa.

Yourpublickeyhasbeensavedin/root/.ssh/id_rsa.pub.

Thekeyfingerprintis:

c2:96:9f:2d:5a:8e:08:42:43:35:2f:85:5e:72:f8:1croot@lab

2)、在备份服务器(192.168.0.200)上创建目录,修改权限,并传送公钥。

[root@lab~]# ssh192.168.0.200"mkdir.ssh;chmod0700.ssh"

Theauthenticityofhost'192.168.0.200(192.168.0.200)'can'tbeestablished.

RSAkeyfingerprintis37:57:55:c1:32:f1:dd:bb:1b:8a:13:6f:89:fb:b8:9d.

Areyousureyouwanttocontinueconnecting(yes/no)? yes

Warning:Permanentlyadded'192.168.0.200'(RSA)tothelistofknownhosts.

root@192.168.0.200'spassword: //输入备份服务器的root密码

[root@lab~]# scp.ssh/id_rsa.pub192.168.0.200:.ssh/authorized_keys2

root@192.168.0.200'spassword:

id_rsa.pub100%2180.2KB/s00:00

3)、测试SSH登录

[root@lab~]# ssh192.168.0.200 //测试SSH登录

Lastlogin:FriNov1610:34:022007from 192.168.0.20

[root@lib~]#

(4)、设置crontab任务,每天执行备份脚本

shell>crontab-e

#每个星期日凌晨3:00执行完全备份脚本

03**0/root/MySQLBackup/mysqlFullBackup.sh>/dev/null2>&1

#周一到周六凌晨3:00做增量备份

03**1-6/root/MySQLBackup/mysqlDailyBackup.sh>/dev/null2>&1

mysqlFullBackup.sh注释:

#!/bin/sh

# Name:mysqlFullBackup.sh

# PS:MySQL DataBase Full Backup.

# Write by:i.Stone

# Last Modify:2007-11-17

#

# Use mysqldump --help get more detail.

#

# 定义变量,请根据具体情况修改

# 定义脚本目录

scriptsDir=`pwd`

# 定义数据库目录

mysqlDir=/usr/local/mysql

# 定义用于备份数据库的用户名和密码

user=root

userPWD=111111

# 定义备份目录

dataBackupDir=/tmp/mysqlbackup

# 定义邮件正文文件

eMailFile=$dataBackupDir/email.txt

# 定义邮件地址

eMail=alter@somode.com

# 定义备份日志文件

logFile=$dataBackupDir/mysqlbackup.log

DATE=`date -I`


echo "" > $eMailFile

echo $(date +"%y-%m-%d %H:%M:%S") >> $eMailFile

cd $dataBackupDir

# 定义备份文件名

dumpFile=mysql_$DATE.sql

GZDumpFile=mysql_$DATE.sql.tar.gz


# 使用mysqldump备份数据库,请根据具体情况设置参数

$mysqlDir/bin/mysqldump -u$user -p$userPWD /

--opt --default-character-set=utf8 --extended-insert=false /

--triggers -R --hex-blob --all-databases /

--flush-logs --delete-master-logs /

--delete-master-logs /

-x > $dumpFile


# 压缩备份文件

if [[ $? == 0 ]]; then

tar czf $GZDumpFile $dumpFile >> $eMailFile 2>&1

echo "BackupFileName:$GZDumpFile" >> $eMailFile

echo "DataBase Backup Success!" >> $eMailFile

rm -f $dumpFile


# Delete daily backup files.

cd $dataBackupDir/daily

rm -f *


# Delete old backup files(mtime>2).

$scriptsDir/rmBackup.sh


# 如果不需要将备份传送到备份服务器或备份服务器为Windows,请将标绿的行注释掉

# Move Backup Files To Backup Server.

#适合Linux(MySQL服务器)到Linux(备份服务器)

$scriptsDir/rsyncBackup.sh

if (( !$? )); then

echo "Move Backup Files To Backup Server Success!" >> $eMailFile

else

echo "Move Backup Files To Backup Server Fail!" >> $eMailFile

fi


else

echo "DataBase Backup Fail!" >> $emailFile

fi

# 写日志文件

echo "--------------------------------------------------------" >> $logFile

cat $eMailFile >> $logFile

# 发送邮件通知

cat $eMailFile | mail -s "MySQL Backup" $eMail

mysqlDailyBackup.sh注释:

#!/bin/sh
# Name:mysqlDailyBackup.sh
# PS:MySQL DataBase Daily Backup.
# Write by:i.Stone
# Last Modify:2007-11-17
#
# 定义变量,请根据具体情况修改
# 定义数据库目录和数据目录
scriptsDir=`pwd`
mysqlDir=/usr/local/mysql
dataDir=$mysqlDir/data
# 定义用于备份数据库的用户名和密码
user=root
userPWD=111111
# 定义备份目录,每日备份文件备份到$dataBackupDir/daily
dataBackupDir=/tmp/mysqlbackup
dailyBackupDir=$dataBackupDir/daily
# 定义邮件正文文件
eMailFile=$dataBackupDir/email.txt
# 定义邮件地址
eMail=alter@somode.com
# 定义日志文件
logFile=$dataBackupDir/mysqlbackup.log
# 得到数据库所在主机的主机名
HOSTNAME=`uname -n`
#
echo "" > $eMailFile
echo $(date +"%y-%m-%d %H:%M:%S") >> $eMailFile
#
# 刷新日志,使数据库使用新的二进制日志文件
$mysqlDir/bin/mysqladmin -u$user -p$userPWD flush-logs
cd $dataDir
# 得到二进制日志列表
fileList=`cat $HOSTNAME-bin.index`
iCounter=0
for file in $fileList
do
iCounter=`expr $iCounter + 1`
done
nextNum=0
iFile=0
for file in $fileList
do
binLogName=`basename $file`
nextNum=`expr $nextNum + 1`
# 跳过最后一个二进制日志(数据库当前使用的二进制日志文件)
if [[ $nextNum == $iCounter ]]; then
echo "Skip lastest!" > /dev/null
else
dest=$dailyBackupDir/$binLogName
# 跳过已经备份的二进制日志文件
if [[ -e $dest ]]; then
echo "Skip exist $binLogName!" > /dev/null
else
# 备份日志文件到备份目录
cp $binLogName $dailyBackupDir
if [[ $? == 0 ]]; then
iFile=`expr $iFile + 1`
echo "$binLogName Backup Success!" >> $eMailFile
fi
fi
fi
done
if [[ $iFile == 0 ]];then
echo "No Binlog Backup!" >> $eMailFile
else
echo "Backup $iFile File(s)." >> $eMailFile
echo "Backup MySQL Binlog OK!" >> $eMailFile

# 如果不需要将备份传送到备份服务器或备份服务器为Windows,请将标绿的行注释掉
# Move Backup Files To Backup Server.
#适合Linux(MySQL服务器)到Linux(备份服务器)

$scriptsDir/rsyncBackup.sh
if [[ $? == 0 ]]; then
echo "Move Backup Files To Backup Server Success!" >> $eMailFile
else
echo "Move Backup Files To Backup Server Fail!" >> $eMailFile
fi
fi
# 发送邮件通知
cat $eMailFile | mail -s "MySQL Backup" $eMail
# 写日志文件
echo "--------------------------------------------------------" >> $logFile
cat $eMailFile >> $logFile

rsyncBackup.sh 注释:

#!/bin/sh
# Name:rsyncBackup.sh
# PS:Move Backup Files To Backup Server.
# Write by:i.Stone
# Last Modify:2007-11-17
#
# 请根据具体情况修改,注意最后有“/
# 定义数据库备份目录
dataBackupDir=/tmp/mysqlbackup/
# 定义备份服务器上存放备份数据的目录
backupServerDir=/root/mysqlbackup/
# 定义备份服务器
backupServer=192.168.0.200
#
# 同步备份文件到备份服务器
rsync -a --delete $dataBackupDir -e ssh $backupServer:$backupServerDir > /dev/null 2>&1

rmBackup.sh注释:

#!/bin/sh
# Name:rmBackup.sh
# PS:Delete old Backup.
# Write by:i.Stone
# Last Modify:2007-11-15
#
# 定义备份目录
dataBackupDir=/tmp/mysqlbackup
# 删除mtime>2 的日志备份文件
find $dataBackupDir -name "mysql_*.gz" -type f -mtime +2 -exec rm {} /; > /dev/null 2>&1

(5) 、恢复数据库到备份时的状态

用mysqldump备份出来的文件是一个可以直接倒入的SQL脚本 , 直接用mysql客户端 导入就可以了。

/usr/local/mysql/bin/mysql-u root -p UserPWD db_name<db_name.sql

对于任何可适用的更新日志,将它们作为mysql的输入:

   %ls-t-r-1HOSTNAME-bin*|xargsmysqlbinlog|mysql-uUser-pUserPWD

ls命令生成更新日志文件的一个单列列表,根据服务器产生它们的次序排序(注意:如果你修改任何一个文件,你将改变排序次序,这将导致更新日志以错误的次序被运用。)

本套备份策略只能恢复数据库到最后一次备份时的状态,要想在崩溃时丢失的数据尽量少应该更频繁的进行备份,要想恢复数据到崩溃时的状态请使用 主从复制机制(replication) 。如果使用本套备份脚本,将日志文件和数据文件放到不同的磁盘上是一个不错的主义,这样不仅可以提高数据写入速度,还能使数据更安全。

?/P>

备份策略四、 使用 主从复制机制(replication)

详情: 点击阅读

脚本下载地址: 点击下载脚本

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值