博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
一个通过BackgroundWorker实现WinForm异步操作的例子
阅读量:6416 次
发布时间:2019-06-23

本文共 4242 字,大约阅读时间需要 14 分钟。

在最近的一个Smart Client项目中,为了演示异步操作的实现,写了一个基于BackgorundWorker的例子。由于这个理基本上实现了BackgorundWorker的大部分功能:异步操作的启动、操作结束后的回调、异步操作的撤销和进度报告等等。尽管没有太多的技术含量,姑且放上来与大家分享。

一、场景描述

下面是程序运行时的截图。本程序模拟这样的一个场景:有两组相互独立的数据需要逐条获取和显示,左边和右边两个groupbox分别代表基于这两组数据的操作,由于他们完全独立,因此可以并行执行。当点击Start按钮,以异步的方式从存储介质中逐条获取数据,并将获取的数据追加到对应的ListBox中,ProgressBar真实反映以获取的数据条数和总记录条数的百分比,同时,当前获取的条数也会在下方的Label上随着操作的继续而动态变化。此外通过点击Stop按钮,可以中止掉当前的操作。当操作被中止后,ProgressBar和Label反映中止的那一刻的状态。

二、代码实现

由于界面上左右两边是两个互不干扰、相互独立的操作,所以分别创建了两个BackgroundWorker组件来负责(如下图:backgroundWorkerLeft和backgroundWorkerRight)。

将两个BackgroundWorker的WorkerReportsProgress和WorkerSupportsCancellation设为true。

我们假设获取的记录数固定,我们为此定义一个常量:

1: private static int MaxRecords = 100;

下面是左边Start按钮的Click event handler:

1: private void buttonStartLeft_Click(object sender, EventArgs e)
2: {
3:     if (this.backgroundWorkerLeft.IsBusy)
4:     {
5:         return;
6:     }
7:     this.listBoxLeft.Items.Clear();
8:     this.backgroundWorkerLeft.RunWorkerAsync(MaxRecords);
9:     this.buttonStartLeft.Enabled = false;
10:     this.buttonCacnelLeft.Enabled = true;
11: }

当Start按钮被点击后,RunWorkerAsync方法被掉调用,我们定义的常量(MaxRecords )当作参数被掺入。随后,将会触发其DoWork事件,Dowork event handler处理代码如下:

1: private void backgroundWorkerLeft_DoWork(object sender, DoWorkEventArgs e)
2: {
3:     try
4:     {
5: e.Result = this.RetrieveData(this.backgroundWorkerLeft, e);
6:     }
7:     catch (Exception ex)
8:     {
9:        MessageBox.Show(ex.Message);
10:        throw;
11:     }
12: }

调用RetrieveData方法逐条获取数据。注意该方法的两个参数:BackgroundWorker和DoWorkEventArgs 对象,返回值是返回数据的数量。由于在buttonStartLeft_Click中,我们将常量MaxRecords 作为参数传入了BackgroundWorker的RunWorkerAsync方法, 此时的e.Argument = MaxRecords。之所以要将这两个参数传入RetrieveData()方法,是因为该方法是为两个BackgroundWorker服务的,需要通过参数来区别当前是哪个BackgroundWorker。我们再来看看RetrieveData方法的定义:

1: private int RetrieveData(BackgroundWorker worker, DoWorkEventArgs e)
2: {
3:     int maxRecords = (int)e.Argument;
4:     int percent = 0;
5:     for (int i = 1; i <= maxRecords; i++)
6:     {
7:         if (worker.CancellationPending)
8:         {
9:             return i;
10:         }
11: 
12:         percent = (int)(((double)i / (double)maxRecords) * 100);
13:         worker.ReportProgress(percent, new KeyValuePair
(i,Guid.NewGuid().ToString()));
14:         Thread.Sleep(100);
15:     }
16: 
17:     return maxRecords;
18: }

通过e.Argument,获得最大数据获取量之后,进行一个for循环,在每次迭代中,如何worker.CancellationPending==true,代表异步操作被显示取消,则直接返回;否则,调用BackgroundWorker的ReportProgress方法。ReportProgress具有两个重载:

  • public void ReportProgress(int percentProgress);
  • public void ReportProgress(int percentProgress, object userState);

percentProgress代表当前进度,从0-100。userState便于传入一些额外的参数。在界面上,由于数据的当前数量需要实时地显示,而记录也是现取现加(取出一条就在ListBox上追加)。所以制定一个KeyValuePair<int,string>对象作为第二个参数。其中Key为当前记录数,Value是一个Guid,代表取出的数据。

ReportProgress的调用将会导致ProgressChanged事件被触发。ProgressChanged event handler用于显示当前进度、当前记录数量和显示获取的纪录:

1: private void backgroundWorkerLeft_ProgressChanged(object sender, ProgressChangedEventArgs e)
2: {
3:     KeyValuePair
record = (KeyValuePair
) e.UserState ;
4:     this.labelResultLeft.Text = string.Format("There are {0} records retrieved!", record.Key);
5:     this.progressBarLeft.Value = e.ProgressPercentage;
6:     this.listBoxLeft.Items.Add(record.Value);
7: }
注:这些操作需要操作UI上的控件,只能在Main Thread中进行。如何在RetrieveData方法进行的话,由于该方式是一个异步方法,是会抛出异常的。

由于操作的时间可能无法预知,在长时间不能完全获取数据的情况下,用户可以需要手工结束掉当前的操作。这个操作实现在Stop按钮的Click事件中:

1: private void buttonCacnelLeft_Click(object sender, EventArgs e)
2: {
3:     this.backgroundWorkerLeft.CancelAsync();
4: }

如何操作正常地结束,BackgroundWorker的RunWorkerCompleted会被触发,下面是RunWorkerCompleted

event handler的定义:

1: private void backgroundWorkerLeft_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
2: {
3:     try
4:     {
5:         this.labelResultLeft.Text = string.Format("Total records: {0}", e.Result);
6:         this.buttonStartLeft.Enabled = true;
7:         this.buttonCacnelLeft.Enabled = false;
8:     }
9:     catch (TargetInvocationException ex)
10:     {
11:         MessageBox.Show(ex.InnerException.GetType().ToString());
12:     }
13: }

上面介绍的是界面左边功能的实现,右边部分的实现完全一致。干兴趣的朋友可以参考Source Code.

分类:
作者:
出处:
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
你可能感兴趣的文章
双NAT
查看>>
我的友情链接
查看>>
firefox using
查看>>
跨域访问-ajax
查看>>
MySQL中UNION
查看>>
5月第二周域名主机网站排名:万网西部数码居前二
查看>>
第三方Jar不打进Apk
查看>>
Qt网络资源汇总(官网、源码、社区、博客)
查看>>
微软企业库研究之日志模块
查看>>
Mysql基本安装及其使用
查看>>
默认虚拟主机&域名跳转301
查看>>
服务器中某些服务无法启动
查看>>
uboot笔记之makefile分析
查看>>
ruby sass 安装
查看>>
高效的远程团队
查看>>
Redis再探
查看>>
使用windows search 搜索文件和文件夹(一)
查看>>
自动备份脚本(批处理)
查看>>
ORACLE 嵌套游标
查看>>
windows 2000 server笔记及安装实验
查看>>