C#数据库备份及还原

1.在用户的配置时,我们需要列出当前局域网内所有的数据库服务器,并且要列出指定服务器的所有数据库,实现代码如下:

取得数据库服务器列表:

  1. public ArrayList GetServerList()
  2. {
  3. ArrayList alServers = new ArrayList() ;
  4. SQLDMO.Application sqlApp = new SQLDMO.ApplicationClass() ;
  5. try
  6. {
  7. SQLDMO.NameList serverList = sqlApp.ListAvailableSQLServers() ;
  8. for(int i = 1;i<= serverList.Count;i++)
  9. {
  10. alServers.Add(serverList.Item(i)) ;
  11. }
  12. }
  13. catch(Exception e)
  14. {
  15. throw(new Exception("取数据库服务器列表出错:"+e.Message)) ;
  16. }
  17. finally
  18. {
  19. sqlApp.Quit() ;
  20. }
  21. return alServers ;
  22. }
 
取得指定数据库服务器的数据库列表   
  1. public ArrayList GetDbList(string strServerName,string strUserName,string strPwd)
  2. {
  3. ServerName = strServerName ;
  4. UserName = strUserName ;
  5. Password = strPwd ;
  6. ArrayList alDbs = new ArrayList() ;
  7. SQLDMO.Application sqlApp = new SQLDMO.ApplicationClass() ;
  8. SQLDMO.SQLServer svr = new SQLDMO.SQLServerClass() ;
  9. try
  10. {
  11. svr.Connect(ServerName,UserName,Password) ;
  12. foreach(SQLDMO.Database db in svr.Databases)
  13. {
  14. if(db.Name!=null)
  15. alDbs.Add(db.Name) ;
  16. }
  17. }
  18. catch(Exception e)
  19. {
  20. throw(new Exception("连接数据库出错:"+e.Message)) ;
  21. }
  22. finally
  23. {
  24. svr.DisConnect() ;
  25. sqlApp.Quit() ;
  26. }
  27. return alDbs ;
  28. }
 

2.数据库的备份和实时进度显示代码:

  1. public bool BackUPDB(string strDbName,string strFileName, ProgressBar pgbMain)
  2. {
  3. PBar = pgbMain ;
  4. SQLDMO.SQLServer svr = new SQLDMO.SQLServerClass() ;
  5. try
  6. {
  7. svr.Connect(ServerName,UserName,Password) ;
  8. SQLDMO.Backup bak = new SQLDMO.BackupClass();
  9. bak.Action =  ;
  10. bak.Initialize = true ;
  11. SQLDMO.BackupSink_PercentCompleteEventHandler pceh =
  12. new SQLDMO.BackupSink_PercentCompleteEventHandler(Step);
  13. bak.PercentComplete += pceh;
  14. bak.Files = strFileName;
  15. bak.Database = strDbName;
  16. bak.SQLBackup(svr);
  17. return true ;
  18. }
  19. catch(Exception err)
  20. {
  21. throw(new Exception("备份数据库失败"+err.Message)) ;
  22. }
  23. finally
  24. {
  25. svr.DisConnect() ;
  26. }
  27. }
  28. private void Step(string message,int percent)
  29. {
  30. PBar.Value = percent ;
  31. }

其中,这两个语句实现了进度的实时显示:

SQLDMO.BackupSink_PercentCompleteEventHandler pceh = new     SQLDMO.BackupSink_PercentCompleteEventHandler(Step); 
    bak.PercentComplete += pceh;

Step就是上面private void Step(string message,int percent) 的方法名称,它用来显示进度条的当前进度。

3.数据库的恢复和杀死进程的代码:

  1. public bool RestoreDB(string strDbName,string strFileName, ProgressBar pgbMain)
  2. {
  3. PBar = pgbMain ;
  4. SQLDMO.SQLServer svr = new SQLDMO.SQLServerClass() ;
  5. try
  6. {
  7. svr.Connect(ServerName,UserName,Password) ;
  8. SQLDMO.QueryResults qr = svr.EnumProcesses(-1) ;
  9. int iColPIDNum = -1 ;
  10. int iColDbName = -1 ;
  11. for(int i=1;i<=qr.Columns;i++)
  12. {
  13. string strName = qr.get_ColumnName(i) ;
  14. if (strName.ToUpper().Trim() == "SPID")
  15. {
  16. iColPIDNum = i ;
  17. }
  18. else if (strName.ToUpper().Trim() == "DBNAME")
  19. {
  20. iColDbName = i ;
  21. }
  22. if (iColPIDNum != -1 && iColDbName != -1)
  23. break ;
  24. }
  25. for(int i=1;i<=qr.Rows;i++)
  26. {
  27. int lPID = qr.GetColumnLong(i,iColPIDNum) ;
  28. string strDBName = qr.GetColumnString(i,iColDbName) ;
  29. if (strDBName.ToUpper() == strDbName.ToUpper())
  30. svr.KillProcess(lPID) ;
  31. }
  32. SQLDMO.Restore res = new SQLDMO.RestoreClass() ;
  33. res.Action = 0 ;
  34. SQLDMO.RestoreSink_PercentCompleteEventHandler pceh =
  35. new SQLDMO.RestoreSink_PercentCompleteEventHandler(Step);
  36. res.PercentComplete += pceh;
  37. res.Files = strFileName ;
  38. res.Database = strDbName ;
  39. res.ReplaceDatabase = true ;
  40. res.SQLRestore(svr) ;
  41. return true ;
  42. }
  43. catch(Exception err)
  44. {
  45. throw(new Exception("恢复数据库失败,请关闭所有和该数据库连接的程序!"+err.Message)) ;
  46. }
  47. finally
  48. {
  49. svr.DisConnect() ;
  50. }
  51. }

其中这个语句取得了所有的进程列表:

SQLDMO.QueryResults qr = svr.EnumProcesses(-1) ;

下面的语句找到和要恢复数据库相关的进程并杀死:

  1. int iColPIDNum = -1 ;
  2. int iColDbName = -1 ;
  3. for(int i=1;i<=qr.Columns;i++)
  4. {
  5. string strName = qr.get_ColumnName(i) ;
  6. if (strName.ToUpper().Trim() == "SPID")
  7. {
  8. iColPIDNum = i ;
  9. }
  10. else if (strName.ToUpper().Trim() == "DBNAME")
  11. {
  12. iColDbName = i ;
  13. }
  14. if (iColPIDNum != -1 && iColDbName != -1)
  15. break ;
  16. }
  17. for(int i=1;i<=qr.Rows;i++)
  18. {
  19. int lPID = qr.GetColumnLong(i,iColPIDNum) ;
  20. string strDBName = qr.GetColumnString(i,iColDbName) ;
  21. if (strDBName.ToUpper() == strDbName.ToUpper())
  22. svr.KillProcess(lPID) ;
  23. }

 C#数据库备份及还原就介绍到这里,希望对你有所帮助。

/////////////////////////////////////////////////////

利用C#还原数据库(SQL SERVER)备份文件到指定路径

利用C#还原数据库(SQL SERVER)备份文件到指定路径

2012-2-19 08:58| 发布者: benben| 查看: 1938| 评论: 0

 
摘要:   最近在做一个数据采集模块,遇到这样一个场景,就是需要将数据库备份文件还原到指定数据库实例下再采集数据。本机测试都没有问题,可一拿到真实环境中测试却发现了一个很大的Bug。所有的数据库都还原不上,很纠 ...
 
 

  最近在做一个数据采集模块,遇到这样一个场景,就是需要将数据库备份文件还原到指定数据库实例下再采集数据。本机测试都没有问题,可一拿到真实环境中测试却发现了一个很大的Bug。所有的数据库都还原不上,很纠结。因为我本以为SQL SERVER 会还原到默认路径下,其实不然。

  当我拿到真实数据库备份文件时,我首先在数据库中运行 restore database RestoreDbName from disk ='H:\DBFolder\Db_Back' ,执行结果如下:

  

  到现在我依然不知道是怎么回事,因为我自己创建的数据库再还原是好使的。怎么会这样?一个意外让我发现原来我建的数据库是在数据库实例默认的路径下,而真实数据库备份文件则不是,它的路径是客户机器上的,肯定不一样。为了验证这个发现,我做了如下的工作。

第一步:创建数据库,create database TestDb 。

第二步:备份数据库。

第三步:直接还原数据库,restore database RestoreDbName from disk ='I:\TestDatabase'  ,执行成功。

第四步:还原数据库,这时候我不是采用默认还原,而是将数据库还原到系统的另一个路径下,代码如下:

restore database RestoreDbName from disk ='I:\TestDatabase'   with  replace, move 'TestDb' to 'H:\DBFolder\RestoreDbName_Data.mdf', move 'TestDb_log' to 'H:\DBFolder\RestoreDbName_Log.ldf'

执行结果:

  第五步:再次备份数据库RestoreDbName。备份后我们用 filelistonly 可以看到当前数据库备份文件的备份路径( restore filelistonly from disk = 'I:\RestoreDb' ), 执行后结果如下:

  从这里我们可以看到当前数据库备份文件的真实物理路径。

第六步:当我再次执行 restore database RestoreDbName from disk ='I:\RestoreDb' ,执行错误。出现了和真实环境一样的错误。原来如上文所述那样,数据库备份文件在还原时,如果当前数据库实例的默认路径和数据库备份文件不相符时需要通过move来解决,而不能单纯的 Restore。

于是我又重新写了一下数据库备份文件还原的相关方法,如下所示。通过这个方法就能简单的实现数据库备份文件还原了。

 /// <summary>
/// 还原数据库文件
/// </summary>
/// <param name="basePath">电子账簿所在根目录</param>
/// <param name="fileName">数据库备份文件名称</param>
/// <param name="databaseName">需要还原的数据库名称</param>
/// <param name="conn">数据库连接</param>
private bool RestoreDataBase(string basePath, string fileName, string databaseName, SqlConnection conn)
{
SqlCommand command = null;
try
{
string restoreStr = string.Empty;
string getLogicFileName = string.Format("restore filelistonly from disk ='{0}'", Path.Combine(basePath, fileName));
DataSet ds = new DataSet();
SqlDataAdapter da = new SqlDataAdapter(getLogicFileName, conn);
da.Fill(ds);
if (ds.Tables == null || ds.Tables[0].Rows.Count == 0) return false; foreach (DataRow dr in ds.Tables[0].Rows)
{
//查找数据库逻辑文件名称
if ("D".Equals(dr["Type"].ToString()))
restoreStr += string.Format("move '{0}' to '{1}',", dr["LogicalName"].ToString(), Path.Combine(basePath, databaseName) + "_Data.mdf");
//查找数据库日志文件名称
else if ("L".Equals(dr["Type"].ToString()))
restoreStr += string.Format("move '{0}' to '{1}',", dr["LogicalName"].ToString(), Path.Combine(basePath, databaseName) + "_Log.ldf");
}
if (string.IsNullOrEmpty(restoreStr))
restoreStr = string.Format("restore database {0} from disk ='{1}'", databaseName, Path.Combine(basePath, fileName));
else
{
restoreStr = string.Format("restore database {0} from disk ='{1}' with replace,", databaseName, Path.Combine(basePath, fileName)) + restoreStr.TrimEnd(',');
}
command = new SqlCommand(restoreStr, conn);
conn.Open();
command.ExecuteNonQuery();
return true;
}
catch (Exception ex)
{
return false;
}
finally
{
conn.Close();
}
}
/// <summary>
/// 还原数据库文件
/// </summary>
/// <param name="basePath">电子账簿所在根目录</param>
/// <param name="fileName">数据库备份文件名称</param>
/// <param name="databaseName">需要还原的数据库名称</param>
/// <param name="conn">数据库连接</param>
private bool RestoreDataBase(string basePath, string fileName, string databaseName, SqlConnection conn)
{
SqlCommand command = null;
try
{
string restoreStr = string.Empty;
string getLogicFileName = string.Format("restore filelistonly from disk ='{0}'", Path.Combine(basePath, fileName));
DataSet ds = new DataSet();
SqlDataAdapter da = new SqlDataAdapter(getLogicFileName, conn);
da.Fill(ds);
if (ds.Tables == null || ds.Tables[0].Rows.Count == 0) return false; foreach (DataRow dr in ds.Tables[0].Rows)
{
//查找数据库逻辑文件名称
if ("D".Equals(dr["Type"].ToString()))
restoreStr += string.Format("move '{0}' to '{1}',", dr["LogicalName"].ToString(), Path.Combine(basePath, databaseName) + "_Data.mdf");
//查找数据库日志文件名称
else if ("L".Equals(dr["Type"].ToString()))
restoreStr += string.Format("move '{0}' to '{1}',", dr["LogicalName"].ToString(), Path.Combine(basePath, databaseName) + "_Log.ldf");
}
if (string.IsNullOrEmpty(restoreStr))
restoreStr = string.Format("restore database {0} from disk ='{1}'", databaseName, Path.Combine(basePath, fileName));
else
{
restoreStr = string.Format("restore database {0} from disk ='{1}' with replace,", databaseName, Path.Combine(basePath, fileName)) + restoreStr.TrimEnd(',');
}
command = new SqlCommand(restoreStr, conn);
conn.Open();
command.ExecuteNonQuery();
return true;
}
catch (Exception ex)
{
return false;
}
finally
{
conn.Close();
}
}

最新文章

  1. replace U to T in mature.fa
  2. 文档:网络通讯包结构(crc校验,加解密)
  3. (转)客户端触发Asp.net中服务端控件事件
  4. Java abstract class 和 interface 的区别
  5. 【poj1741】 Tree
  6. UVA5870 乱搞 Smooth Visualization
  7. sublime text2支持ng
  8. [转]每天一个linux命令目录
  9. .NET文件上传的大小限制配置
  10. 简短介绍 C# 6 的新特性 .net 6都要来了?
  11. SignalR发布后不能生成signalr/hubs
  12. 利用COM组件IPicture读取jpg、gif、bmp图片文件数据和显示图片
  13. 用python写爬虫
  14. Zend Studio 无法打开的解决办法
  15. 使用Visual Studio 2017作为Linux C++开发工具
  16. Apache优化配置:修改最大并发连接数
  17. SpringBoot整合Netty并使用Protobuf进行数据传输(附工程)
  18. JZ2440 裸机驱动 第6章 存储控制器
  19. 各种”xxx“ native gem required installed build tools 报错
  20. HTML5-入门3。

热门文章

  1. 2017-5新版ionic3.1 新命令及一些常用命令
  2. css3背景渐变色代码
  3. printFinal用法示例
  4. 【译】x86程序员手册01
  5. TriAquae 是一款由国产的基于Python开发的开源批量部署管理工具
  6. mysqlconnector将EXCEL表数据导入数据库
  7. eclipse常用设置之项目分组查看
  8. vue刷新本页面
  9. IE浏览器new Date()带参返回NaN解决方法
  10. webpack-dev-middleware 与 webpack-hot-middlware