c# 實現資料庫的備份和恢復
在我們做資料庫系統的程式時,經常需要為客戶做一個資料庫的備份和恢復程式,特別是對於一些非專業的資料庫使用者,這個程式更是必不可少,而且操作必需足夠簡單。因為在很多系統中,資料庫的備份恢復功能都是相近的,因此,我們最好做一個通用的資料庫備份恢復程式,這樣就不必每個系統都開發一套了。
要開發這樣一套系統,我個人認為應該滿足以下要求:
1. 備份恢復操作應該有歷史記錄(必需有一個備份列表,列出備份檔案的相關資訊),便於使用者查詢以往的備份。
2. 對於每一次備份和恢復應允許使用者記錄備份和恢復的原因。
3. 系統應允許使用者進行簡單的配置,並且配置可以儲存。
4. 備份和恢復應該足夠簡單,最好類似與檔案的複製,對於已經進行的備份,應允許使用者從備份列表恢復。
5. 即使因為某種原因使要恢復的資料庫正在佔用,也應該允許使用者恢復(這一點很重要,因為你不能指望使用者自己保證資料庫的獨佔性)。
6. 實時顯示備份或恢復的當前進度。
要達到以上的要求,我想我們應該這樣設計系統:
1. 對於每一次的資料庫備份和恢復,我們都記下當時的資料庫伺服器名,資料庫名,備份檔案全路徑名,備份時間,進行備份或恢復的原因等資訊,並把這些資訊以XML的形式儲存,下面是我得一個備份檔案例項:
2. 對於使用者的配置可以這樣進行:
上面依次記錄了備份程式的名稱(顯示在備份窗體的標題欄,無實際用處),備份歷史最大記錄數,備份的資料庫伺服器名稱,備份的資料庫名稱,使用者名稱,密碼(已經過加密)等資訊。我們在程式剛開始啟動時,就自動把這些資訊應用到使用者介面上去,這樣就不用使用者重新設定了。
3. 備份時我們採取直接備份到檔案的方法,使用者只需使用儲存檔案對話方塊指定要備份的位置和檔名即可,其餘的工作通過程式完成,恢復也一樣,只需通過開啟檔案對話方塊指定從其中恢復的檔案即可。
4. 我們在恢復時,先殺死要恢復的資料庫所關聯的所有使用者執行緒,然後再進行恢復,這樣就不會存在因為資料庫獨佔性引起的恢復錯誤。
5. 對於實時顯示備份和恢復的進度問題,我們採取SQL-DMO的回撥函式的方式實現。
下面是相關技術難點的程式碼實現(因為個人的喜好,在此已C#的程式碼形式實現):
1. 在使用者的配置時,我們需要列出當前區域網內所有的資料庫伺服器,並且要列出指定伺服器的所有資料庫,實現程式碼如下:
取得資料庫伺服器列表:
public ArrayList GetServerList()
{
ArrayList alServers = new ArrayList() ;
SQLDMO.Application sqlApp = new SQLDMO.ApplicationClass() ;
try
{
SQLDMO.NameList serverList = sqlApp.ListAvailableSQLServers() ;
for(int i = 1;i<= serverList.Count;i++)
{
alServers.Add(serverList.Item(i)) ;
}
}
catch(Exception e)
{
throw(new Exception("取資料庫伺服器列表出錯:"+e.Message)) ;
}
finally
{
sqlApp.Quit() ;
}
return alServers ;
}
取得指定資料庫伺服器的資料庫列表
public ArrayList GetDbList(string strServerName,string strUserName,string strPwd)
{
ServerName = strServerName ;
UserName = strUserName ;
Password = strPwd ;
ArrayList alDbs = new ArrayList() ;
SQLDMO.Application sqlApp = new SQLDMO.ApplicationClass() ;
SQLDMO.SQLServer svr = new SQLDMO.SQLServerClass() ;
try
{
svr.Connect(ServerName,UserName,Password) ;
foreach(SQLDMO.Database db in svr.Databases)
{
if(db.Name!=null)
alDbs.Add(db.Name) ;
}
}
catch(Exception e)
{
throw(new Exception("連線資料庫出錯:"+e.Message)) ;
}
finally
{
svr.DisConnect() ;
sqlApp.Quit() ;
}
return alDbs ;
}
2. 資料庫的備份和實時進度顯示程式碼:
public bool BackUPDB(string strDbName,string strFileName, ProgressBar pgbMain)
{
PBar = pgbMain ;
SQLDMO.SQLServer svr = new SQLDMO.SQLServerClass() ;
try
{
svr.Connect(ServerName,UserName,Password) ;
SQLDMO.Backup bak = new SQLDMO.BackupClass();
bak.Action = 0 ;
bak.Initialize = true ;
SQLDMO.BackupSink_PercentCompleteEventHandler pceh = new SQLDMO.BackupSink_PercentCompleteEventHandler(Step);
bak.PercentComplete += pceh;
bak.Files = strFileName;
bak.Database = strDbName;
bak.SQLBackup(svr);
return true ;
}
catch(Exception err)
{
throw(new Exception("備份資料庫失敗"+err.Message)) ;
}
finally
{
svr.DisConnect() ;
}
}
private void Step(string message,int percent)
{
PBar.Value = percent ;
}
其中,這兩個語句實現了進度的實時顯示:
SQLDMO.BackupSink_PercentCompleteEventHandler pceh = new SQLDMO.BackupSink_PercentCompleteEventHandler(Step);
bak.PercentComplete += pceh;
Step就是上面private void Step(string message,int percent) 的方法名稱,它用來顯示進度條的當前進度。
3. 資料庫的恢復和殺死程式的程式碼:
public bool RestoreDB(string strDbName,string strFileName, ProgressBar pgbMain)
{
PBar = pgbMain ;
SQLDMO.SQLServer svr = new SQLDMO.SQLServerClass() ;
try
{
svr.Connect(ServerName,UserName,Password) ;
SQLDMO.QueryResults qr = svr.EnumProcesses(-1) ;
int iColPIDNum = -1 ;
int iColDbName = -1 ;
for(int i=1;i<=qr.Columns;i++)
{
string strName = qr.get_ColumnName(i) ;
if (strName.ToUpper().Trim() == "SPID")
{
iColPIDNum = i ;
}
else if (strName.ToUpper().Trim() == "DBNAME")
{
iColDbName = i ;
}
if (iColPIDNum != -1 && iColDbName != -1)
break ;
}
for(int i=1;i<=qr.Rows;i++)
{
int lPID = qr.GetColumnLong(i,iColPIDNum) ;
string strDBName = qr.GetColumnString(i,iColDbName) ;
if (strDBName.ToUpper() == strDbName.ToUpper())
svr.KillProcess(lPID) ;
}
SQLDMO.Restore res = new SQLDMO.RestoreClass() ;
res.Action = 0 ;
SQLDMO.RestoreSink_PercentCompleteEventHandler pceh = new SQLDMO.RestoreSink_PercentCompleteEventHandler(Step);
res.PercentComplete += pceh;
res.Files = strFileName ;
res.Database = strDbName ;
res.ReplaceDatabase = true ;
res.SQLRestore(svr) ;
return true ;
}
catch(Exception err)
{
throw(new Exception("恢復資料庫失敗,請關閉所有和該資料庫連線的程式!"+err.Message)) ;
}
finally
{
svr.DisConnect() ;
}
}
其中這個語句取得了所有的程式列表:
SQLDMO.QueryResults qr = svr.EnumProcesses(-1) ;
下面的語句找到和要恢復資料庫相關的程式並殺死:
int iColPIDNum = -1 ;
int iColDbName = -1 ;
for(int i=1;i<=qr.Columns;i++)
{
string strName = qr.get_ColumnName(i) ;
if (strName.ToUpper().Trim() == "SPID")
{
iColPIDNum = i ;
}
else if (strName.ToUpper().Trim() == "DBNAME")
{
iColDbName = i ;
}
if (iColPIDNum != -1 && iColDbName != -1)
break ;
}
for(int i=1;i<=qr.Rows;i++)
{
int lPID = qr.GetColumnLong(i,iColPIDNum) ;
string strDBName = qr.GetColumnString(i,iColDbName) ;
if (strDBName.ToUpper() == strDbName.ToUpper())
svr.KillProcess(lPID) ;
}
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/554469/viewspace-914957/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Xtrabackup實現資料庫備份和災難恢復資料庫
- 資料庫資料的恢復和備份資料庫
- 【備份恢復】從備份恢復資料庫資料庫
- PostgreSql資料庫的備份和恢復SQL資料庫
- 備份和恢復postgreSQL資料庫SQL資料庫
- Mongo 資料庫備份和恢復命令Go資料庫
- 備份和恢復SQL Server資料庫SQLServer資料庫
- 備份與恢復:polardb資料庫備份與恢復資料庫
- 資料庫的備份與恢復資料庫
- DB2資料庫的備份和恢復DB2資料庫
- oracle資料庫備份和恢復的內容Oracle資料庫
- 使用Mysqldump備份和恢復MySQL資料庫MySql資料庫
- 非RMAN熱備份資料庫和恢復資料庫
- db2備份和恢復資料庫DB2資料庫
- 【備份恢復】Oracle 資料備份與恢復微實踐Oracle
- 【備份恢復】noarchive模式下使用增量備份恢復資料庫Hive模式資料庫
- mysql的資料庫備份與恢復MySql資料庫
- oracle資料庫的備份與恢復Oracle資料庫
- 【備份恢復】在 ARCHIVELOG 模式下執行資料庫還原和恢復操作(源庫備份源庫恢復)Hive模式資料庫
- 可定製的資料庫備份和恢復程式資料庫
- SqlServer資料庫恢復備份資料的方法SQLServer資料庫
- 達夢資料庫備份恢復資料庫
- postgresql備份與恢復資料庫SQL資料庫
- mongo資料庫備份與恢復Go資料庫
- Informix資料庫備份與恢復ORM資料庫
- Mysql資料庫備份及恢復MySql資料庫
- SQL Server 資料庫備份還原和資料恢復SQLServer資料庫資料恢復
- 求助:hibernate如何實現資料庫備份與恢復啊?資料庫
- NoSQL 資料庫案例實戰 -- MongoDB資料備份、恢復SQL資料庫MongoDB
- 2 Day DBA-管理方案物件-執行備份和恢復-資料庫備份和恢復概念物件資料庫
- java中實現MYSQL的備份和恢復JavaMySql
- 關閉資料庫的備份與恢復資料庫
- Oracle資料庫的備份與恢復(轉)Oracle資料庫
- Oracle 資料庫的備份與恢復(轉)Oracle資料庫
- rman資料庫全庫備份與恢復資料庫
- 2 Day DBA-管理方案物件-執行備份和恢復-資料庫備份和恢復概覽物件資料庫
- RMAN備份恢復——RAC環境資料庫的備份(zt)資料庫
- RMAN備份恢復--RAC環境資料庫的備份(十)資料庫