c# 實現資料庫的備份和恢復

binggeixu發表於2007-05-14
摘至csdn中kv4000的.


在我們做資料庫系統的程式時,經常需要為客戶做一個資料庫的備份和恢復程式,特別是對於一些非專業的資料庫使用者,這個程式更是必不可少,而且操作必需足夠簡單。因為在很多系統中,資料庫的備份恢復功能都是相近的,因此,我們最好做一個通用的資料庫備份恢復程式,這樣就不必每個系統都開發一套了。

要開發這樣一套系統,我個人認為應該滿足以下要求:

1. 備份恢復操作應該有歷史記錄(必需有一個備份列表,列出備份檔案的相關資訊),便於使用者查詢以往的備份。

2. 對於每一次備份和恢復應允許使用者記錄備份和恢復的原因。

3. 系統應允許使用者進行簡單的配置,並且配置可以儲存。

4. 備份和恢復應該足夠簡單,最好類似與檔案的複製,對於已經進行的備份,應允許使用者從備份列表恢復。

5. 即使因為某種原因使要恢復的資料庫正在佔用,也應該允許使用者恢復(這一點很重要,因為你不能指望使用者自己保證資料庫的獨佔性)。

6. 實時顯示備份或恢復的當前進度。

要達到以上的要求,我想我們應該這樣設計系統:

1. 對於每一次的資料庫備份和恢復,我們都記下當時的資料庫伺服器名,資料庫名,備份檔案全路徑名,備份時間,進行備份或恢復的原因等資訊,並把這些資訊以XML的形式儲存,下面是我得一個備份檔案例項:






hrmjx4
測試備份
E:ricatex2003.bak



hrmjx4
每週例行備份
E:hrmjx4040205.bak





2. 對於使用者的配置可以這樣進行:




資料庫備份
20
.
book
sa
2iUc94tkpsg=


上面依次記錄了備份程式的名稱(顯示在備份窗體的標題欄,無實際用處),備份歷史最大記錄數,備份的資料庫伺服器名稱,備份的資料庫名稱,使用者名稱,密碼(已經過加密)等資訊。我們在程式剛開始啟動時,就自動把這些資訊應用到使用者介面上去,這樣就不用使用者重新設定了。

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) ;
}

[@more@]

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/554469/viewspace-914957/,如需轉載,請註明出處,否則將追究法律責任。

相關文章