异步备份和还原数据库:.NET发现之旅(六)
信息系统是数据密集型的,数据的套帐,备份,还原是客户最希望有的功能,这一节课就讨论下C/S系统下数据库的异步备份和还原,B/S系统的数据备份和还原和这个类似。
既然是异步,首先会想到使用多线程技术。.NET平台提供了一整套的线程处理技术,使用线程的好处是,可以让一个线程做一件事情,多个线程之间根据时间片机制抢夺CPU和I/O资源,UI线程用于绘制界面,保证界面永远对客户的响应,而工作线程用于计算工作。
除了从头开发线程外,.NET也提供了一个封装好的线程组件BackgroundWorker,该组件让您能够在应用程序的主要 UI线程以外的其他线程上异步(“在后台”)执行耗时的操作。比如耗时耗资源的常用操作如下:
· 图像下载
· Web 服务调用
· 文件下载和上载(包括点对点应用程序)
· 复杂的本地计算
· 数据库事务
· 本地磁盘访问(相对于内存访问来说其速度很慢)
类似这样的操作可能导致用户界面在操作运行时挂起。如果需要用户界面的响应却遇到与此类操作关联的长时间延迟,BackgroundWorker组件可以提供一种方便的解决方案。若要使用 BackgroundWorker,只需要告诉该组件要在后台执行的耗时的辅助方法,然后调用 RunWorkerAsync 方法。在辅助方法以异步方式运行的同时,您的调用线程继续正常运行。该方法运行完毕,BackgroundWorker激发 RunWorkerCompleted 事件(可选择包含操作结果)向调用线程发出警报。
“组件”选项卡的“工具箱”中提供了 BackgroundWorker组件。VS2005,VS2008, VS2010都有这个组件,如下图:
若要向窗体添加 BackgroundWorker,请将 BackgroundWorker组件拖到窗体上即可。
若要启动异步操作,请使用 RunWorkerAsync 方法。RunWorkerAsync 采用一个可选的 object 参数,可以使用该参数将变量传递给辅助方法。BackgroundWorker类公开 DoWork 事件,您的辅助线程通过 DoWork 事件处理程序附加到该事件。
BackgroundWorker包含三个主要的事件:
1,DoWork 事件
调用 RunWorkerAsync 方法时将引发此事件。在此,您就可以启动操作来执行可能很耗时的工作。
2, RunWorkerCompleted 事件
DoWork事件处理程序返回时将引发此事件。如果操作成功完成,并且其结果在 DoWork事件处理程序中进行了分配,则可以通过RunWorkerCompletedEventArgs.Result属性访问该结果。
3, ProgressChanged 事件
调用 ReportProgress 方法时将引发此事件。ProgressChanged事件向用户报告异步操作的进度
熟悉了组建用法后,开始做异步的数据库备份和还原
一,异步数据库备份,界面如下:
点击开始备份按钮,选择一个文件夹保存备份文件
然后开始备份,线程会不停的汇报它自己执行的进度,用一个进度条来接受线程的进度,这个是通过ProgressChanged事件完成的。
当备份完成后,线程会通知系统,它已经返回。如果中间线程出现执行错误,通过程序来捕获。这个是通过RunWorkerCompleted 事件来完成的,而备份任务是通过DoWork 事件来完成的。
异步备份数据库的代码如下:
- //启动备份
- private void btnBackUp_Click(object sender, EventArgs e)
- {
- try
- {
- SaveFileDialog sfd = new SaveFileDialog();
- sfd.Title = "请选择备份保存的目录";
- sfd.Filter = "(*.Bak)|*.Bak;|(All Files)|*.*";
- if (sfd.ShowDialog() == DialogResult.OK && sfd.FileName != string.Empty)
- {
- string filePath = sfd.FileName;
- //启动异步备份,开始执行DoWork事件
- this.backgroundWorker1.RunWorkerAsync(new string[] { filePath, "InstallationsManager" });
- }
- }
- catch (System.Exception ex)
- {
- MessageBox.Show("异常:" + ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- }
- //备份数据库
- private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
- {
- try
- {
- using (SqlConnection con = new SqlConnection(ConfigurationManager.ConnectionStrings["LongXiConnectionString"].ConnectionString))
- {
- con.Open();
- //汇报进度
- this.backgroundWorker1.ReportProgress(20);
- string[] args = (string[])e.Argument;
- string filePath = args[0];
- SqlCommand cmd = new SqlCommand();
- cmd.Connection = con;
- //备份数据库
- cmd.CommandText = "use master;backup database " + args[1] + " to disk = '" + filePath + "' ";
- cmd.ExecuteNonQuery();
- this.backgroundWorker1.ReportProgress(60);
- //线程完成时返回的信息
- e.Result = "备份数据成功!";
- //this.label5.Text = "aaaaaaaa"; //Control.CheckForIllegalCrossThreadCalls = false
- this.backgroundWorker1.ReportProgress(100);
- }
- }
- catch (System.Exception ex)
- {
- MessageBox.Show("异常:" + ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- }
- //线程执行完成后的收尾工作
- private void backgroundWorker1_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
- {
- try
- {
- if (e.Error != null)
- {
- MessageBox.Show("异常:" + e.Error.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- else if (e.Cancelled)
- {
- MessageBox.Show("线程已经退出!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
- }
- else
- {
- //MessageBox.Show(e.Result.ToString(), "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
- //向UI提示备份数据库成功
- this.lblInfo.Text = e.Result.ToString();
- }
- }
- catch (System.Exception ex)
- {
- MessageBox.Show("异常:" + ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- }
- //向UI线程汇报进度,使进度条的值增加
- private void backgroundWorker1_ProgressChanged(object sender, ProgressChangedEventArgs e)
- {
- try
- {
- this.pbarBackUp.Value = e.ProgressPercentage;
- }
- catch (System.Exception ex)
- {
- MessageBox.Show("异常:" + ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- }
二,异步还原数据库
和备份类似,这里不再讲解,代码如下:
- //启动还原
- private void btnRestore_Click(object sender, EventArgs e)
- {
- //第一次后悔药
- if (MessageBox.Show("该操作将数据覆盖!如果选择[是],将原来的数据覆盖;如果选择[否],将取消恢复.", "提示", MessageBoxButtons.YesNo, MessageBoxIcon.Question) == DialogResult.No)
- return;
- //第二次后悔药
- if (MessageBox.Show("请再次确认,该操作不能恢复!如果选择[是],将原来的数据覆盖;如果选择[否],将取消恢复.", "提示", MessageBoxButtons.YesNo, MessageBoxIcon.Warning) == DialogResult.No)
- return;
- //没得后悔药吃了,开始执行还原
- try
- {
- OpenFileDialog ofd = new OpenFileDialog();
- ofd.Title = "请选择要恢复的备份文件";
- ofd.Filter = "(*.Bak)|*.Bak;|(All Files)|*.*";
- if (ofd.ShowDialog() == DialogResult.OK && ofd.FileName != string.Empty)
- {
- string filePath = ofd.FileName;
- //启动异步还原,开始执行DoWork事件
- this.backgroundWorker1.RunWorkerAsync(new string[] { filePath, "InstallationsManager" });
- }
- }
- catch (System.Exception ex)
- {
- MessageBox.Show("异常:" + ex.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- }
- //还原数据库
- private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
- {
- this.backgroundWorker1.ReportProgress(30);
- using (SqlConnection con = new SqlConnection(ConfigurationManager.ConnectionStrings["LongXiConnectionString"].ConnectionString))
- {
- con.Open();
- //汇报进度
- this.backgroundWorker1.ReportProgress(20);
- Thread.Sleep(1000);
- string[] args = (string[])e.Argument;
- string filePath = args[0];
- SqlCommand cmd = new SqlCommand();
- this.backgroundWorker1.ReportProgress(30);
- Thread.Sleep(1000);
- this.backgroundWorker1.ReportProgress(60);
- cmd.Connection = con;
- //还原数据库
- cmd.CommandText = "use master;alter database " + args[1] + " set offline with rollback immediate;restore database " + args[1] + " from disk = '" + filePath + "' with replace";
- cmd.ExecuteNonQuery();
- this.backgroundWorker1.ReportProgress(80);
- Thread.Sleep(1000);
- this.backgroundWorker1.ReportProgress(100);
- //线程完成时返回的信息
- e.Result = "恢复数据成功!";
- }
- }
- //汇报进度,使进度条的值增加
- private void backgroundWorker1_ProgressChanged(object sender, ProgressChangedEventArgs e)
- {
- this.pbarRestore.Value = e.ProgressPercentage;
- }
- //线程执行完成后的收尾工作
- private void backgroundWorker1_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
- {
- if (e.Error != null)
- {
- MessageBox.Show("异常:" + e.Error.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error);
- }
- else if (e.Cancelled)
- {
- MessageBox.Show("线程已经退出!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
- }
- else
- {
- //MessageBox.Show(e.Result.ToString(), "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
- //向UI提示恢复数据库成功
- this.lblInfo.Text = e.Result.ToString();
- }
- }
本文转自terryli51CTO博客,原文链接:http://blog.51cto.com/terryli/335062 ,如需转载请自行联系原作者
相关文章:

看动画轻松理解「链表」实现「LRU缓存淘汰算法」
作者 | 程序员小吴,哈工大学渣,目前正在学算法,开源项目 「 LeetCodeAnimation 」5500star,GitHub Trending 榜连续一月第一。本文为 AI科技大本营投稿文章(欢迎给我们投稿,投稿请联系微信1092722531&#…

东北师大计算机考研报名人数,东北师范大学考研难吗?一般要什么水平才可以进入?...
问:从东北师范大学毕业的学生就业怎么样?值不值得报考?答:想要了解东北师范大学更多毕业生就业情况见>>>东北师范大学总之,东北师范大学就业率相对来说是比较良好的,如果大家对此学校感兴趣的话&a…

2009-徘徊-开场白
徘徊 不知道该怎么走coding已经随风散去思想不复存在,9月应该是个很好的月份。步入一个公司,当时的部长很强,虽然是ASP招我进来5天,走了。进入后的第一个是做一个RPGmaker的游戏coding 没什么关系08年刚开始似乎就很少编码了窝在宿…

未获得计算机访问权限,如何获取文件夹的访问权限
有些系统文件夹打不开,显示信息“拒绝你访问该文件夹”,有点让人摸不着头脑,明明我是管理员账号,明明整台电脑都是我的,你凭什么不让我访问呢,原来系统内是有比较复杂的权限分配的,我们可以稍作…

生成验证码点击可刷新
我把生成验证码与生成验证码图片封装成一个静态方法,放到ValidateCode类里 /// <summary> /// 生成验证码的类 /// </summary> public static class ValidateCode { /// <summary> /// 生成验证码 /// </summary> ///…

redhat enterprise linux 下配置本地yum源
一、在linux 6.1中本地yum源配置:首先编辑yum源配置文件我们可以再这个目录中新创建一个配置文件, #cd /etc/yum.repos.d, #vim yum.repo配置文件内容简介:[ ]内的是仓库的名字 name是仓库的描述也可以说是名字 baseurl 仓库的位置 enabled…

2018年最后几天学什么?给你关注度最高的10篇文章
作者 | Mybridge译者 | Linstancy整理 | Jane出品 | AI科技大本营【导语】我们从 12 月里近1400篇机器学习文章进行了排名,并挑选出最受大家关注的十篇文章。这些文章的内容主要是由 Google、DeepMind、OpenAI 等科技公司发布的自家在机器学习领域最新技术研究&…

详解Silverlight Treeview的HierarchicalDataTemplate使用
在Silverlight项目中,Treeview控件是比较常用的表示层次或者等级的控件,该控件可以非常清晰的显示数据之间的隶属关系。对于Treeview控件的基本使用已经有很多文章介绍,这里我想讲解一下Silverlight Treeivew的HierarchicalDataTemplate的使用…

win7计算机组策略打不开,win7系统gpedit.msc组策略打不开该怎么解决?方法步骤
电脑作为一款日常使用工具,用的时间长了,就会遇到各种事情,譬如win7系统gpedit.msc组策略打不开的状况,不用猜,正常情况下大伙第一次碰到win7系统gpedit.msc组策略打不开问题的肯定都会蒙圈,其实要完美解决…

免费公开课报名 | 达观数据个性化推荐系统实践
在人工智能浪潮之下,个性化推荐技术更是风靡业界,在金融、传媒、短视频、电商、教育等诸多领域大放异彩,影响人们生活的方方面面。不仅方便了人们获取各种各样的信息,还给企业带来了收益的大幅提升。本次公开课我们邀请到了达观数…
串口服务器支持多台上位机,RS485多机通信一台上位机两台下位机问题,
RS485一主机两个下位机,上位机通过发送下位机地址查询下位机两个按钮的状态,仿真时同时在P1口和P2口显示按钮状态一、原理图(原文件名:1.jpg)二、程序1、上位机程序#include "main_host.h"/* 延时t毫秒 */void delay(uint t){uint i;while(t--…

印度小哥“神剑”:PDF提取表格so easy!
作者 | 若名出品 | AI科技大本营如果经常跟数据表格打交道,那你应该体验过那种令人烦躁到抓狂的心情。但现在,学会下面将要介绍的一款工具的使用方法,相信我,它会让你在工作中简直不能更舒爽。Excalibur,从古希腊语翻译…

Linux系统中的Page cache和Buffer cache
Free命令显示内存 首先,我们来了解下内存的使用情况: Mem:表示物理内存统计 total:表示物理内存总量(total used free) used:表示总计分配给缓存(包含buffers 与cache )使用的数量,…

Java垃圾回收调优
在Java中,通常通讯类型的服务器对GC(Garbage Collection)比较敏感。通常通讯服务器每秒需要处理大量进出的数据包,需要解析,分解成不同的业务逻辑对象并做相关的业务处理,这样会导致大量的临时对象被创建和回收。同时服务器如果需…

高性能的MySQL(6)查询慢与重构查询
只有好的库表结构、合理的索引还不够,我们还需要合理的设计查询,齐头并进,一个不少才能充分发挥MySQL的优势。 一、查询为什么会慢? 每一个查询由一系列的子任务组成,每个子任务都会消耗一定的时间。这个我们在之前的单…

wamp找不到服务器,WampServer服务器多站点配置后打不开phpMyAdmin的解决办法
WampServer服务器多站点配置后打不开phpMyAdmin的解决办法 (Forbidden You dont have permission to access / on this server.解决方法)1、首先找到wampserver安装目录下面的httpd.conf配置文件 文件路径:电脑磁盘:\wamp\bin\apache\Apache2.2.21\conf\httpd.conf打…

TSM简介(一)- 原理与特点
IBM的TSM软件是市场上主流的企业备份解决方案。它可以通过网络或者SAN集中备份企业中几乎所有系统平台的数据到磁带机或者磁带库。它可以对大多数数据,如ORACLE, SQL SERVER, DB2, LOTUS DOMINO进行在线备份,也是唯一可以对AS/400数据进行网络备份的解决…

AI人工智能的未来?AI科技与AI教程?答案都在这里!
欢迎大家关注CSDN旗下专业的AI平台【AI科技大本营】,AI科技大本营公众号为广大读者提供中国AI技术分析、学习AI技术、AI技术应用和AI科技资讯等服务。 AI科技大本营是中国专业IT社区CSDN旗下的AI垂直媒体,致力于关注并报道全球人工智能领域技术及产业方面…

Linux命令:文本处理工具awk详解
awk命令简介: awk是一个强大的文本分析工具,通常,awk是以文件的每一行,为处理单位的。awk每接收文件的一行,然后执行相应的命令,来处理文本。 1.命令格式: awk pattern {action}{filenames} 其中 pattern 表…

ForefrontTMG关于单一的网络适配器限制
来自于ISA2008的帮助文件.Microsoft Forefront Threat Management Gateway 可以安装在具有单一网络适配器的计算机上。 通常,当 Forefront TMG 位于公司内部网络或位于外围网络而网络边缘有另一防火墙时使用此配置,防止公司资源受到来自 Internet 的***。…

ajax的auto是true,一个AJAX自动完成功能的js封装源码[支持中文]
ScreenshotsPaginationInternationalizationRich formatText selection源码下载有个问题,等高手解决,就是怎么搞都不能支持中文,不知道要改哪个地方!期待比如 new CAPXOUS.AutoComplete("wiki", function() {return "autocomplete3.asp?typing" …

非计算机专业如何转行AI,找到算法offer?
作者 | Nick-Atom 责编 | 琥珀 【AI科技大本营导读】目前,各行业都在尝试着用机器学习/深度学习来解决自身行业的需求。在这个过程中,最为稀缺的也是高质量人工智能人才。 这一年我们见证了不断有非计算机专业学生转行人工智能的现象,每个想…

Swing布局管理器介绍
当选择使用JPanel和顶层容器的content pane时,需要考虑布局管理。JPanel缺省是初始化一个FlowLayout,而content pane缺省是初始化一个BorderLayout。下面将分别介绍几种最常用的布局管理器:FlowLayout、BorderLayout、BoxLayout、CardLayout、…

圣诞节!教你用Python画棵圣诞树
作者 | 糖甜甜甜,985高校经管研二,擅长用 Python、R、tableau 等工具结合统计学和机器学习模型做数据分析。来源 | 经管人学数据分析(ID:DAT-2017)如何用Python画一个圣诞树呢?最简单:1height 523stars 1…

给动态生成的按钮添加ajax,Ajax/Javascript动态创建按钮的问题
你没有指定一个事件处理程序,您调用它。函数createButtons马上被调用,并且它返回的任何东西都被赋值给window.onload。window.onload createButtons();必须window.onload createButtons;你必须是你正在使用domready中和的onload另一个问题。两个不同的…

Android进阶笔记:Messenger源码详解
Messenger可以理解为一个是用于发送消息的一个类用法也很多,这里主要分析一下再跨进程的情况下Messenger的实现流程与源码分析。相信结合前面两篇关于aidl解析文章能够更好的对aidl有一个认识。(Android进阶笔记:AIDL内部实现详解 ࿰…

程序员春运抢票的正确姿势!
作者 | 屠敏出品 | CSDN资讯「有钱没钱,回家过年。」转瞬间,2018 年余额已严重不足,而 2019 年还正在赶来的路上:根据国家法定假日规定,距离 2019 年春节( 2019 年 2 月 5 日)的到来还有 45 天&…

centos 6.8 编译安装git 2.11.0
系统环境:CentOS release 6.8 (Final) 默认Git :1.7.1 需求git :2.11 卸载centos自带的git:yum remove git -y 下载git-2.11.0.tar.gz 上传至服务器,下载链接:http://distfiles.macports.org/git/ 解压安装…

小黑盒不显示服务器,steam上买的游戏小黑盒不显示 | 手游网游页游攻略大全
发布时间:2018-04-18贪吃蛇遇上打方块是一款最近非常热门的休闲小游戏,游戏将贪吃蛇和打方块合而为一吸引了很多玩家!但是好多小伙们都不知道该怎么玩,下面小编来教你一个快速入门的小技巧吧!~ 小技巧: 1.通过吃游戏带数字的黄色圆点,增加自己的长 ...标签ÿ…

基于架构的上网行为管理产品界面对比
当前上网行为管理产品风靡市场,对用户的网络管理的能力有很大的提高,给用户带来便利的同时也给用户带来很多困惑。目前市场上的上网行为管理产品在功能点上基本相同,于是一些设备的厂商开始关注设备的其它附加功能,比如网络管理人…