首页 > 代码库 > winform - BackgroundWorker
winform - BackgroundWorker
http://www.cnblogs.com/happy555/archive/2007/11/07/952315.html
在VS2005中添加了BackgroundWorker组件,该组件在多线程编程方面使用起来非常方便,然而在开始时由于没有搞清楚它的使用机制,走了不少的弯路,现在把我在使用它的过程中的经验与诸位分享一下。
BackgroundWorker类中主要用到的有这列属性、方法和事件:
重要属性:
1、CancellationPending 获取一个值,指示应用程序是否已请求取消后台操作。通过在DoWork事件中判断CancellationPending属性可以认定是否需要取消后台操作(也就是结束线程);
2、IsBusy 获取一个值,指示 BackgroundWorker 是否正在运行异步操作。程序中使用IsBusy属性用来确定后台操作是否正在使用中;
3、WorkerReportsProgress 获取或设置一个值,该值指示BackgroundWorker能否报告进度更新
4、WorkerSupportsCancellation 获取或设置一个值,该值指示 BackgroundWorker 是否支持异步取消。设置WorkerSupportsCancellation为true使得程序可以调用CancelAsync方法提交终止挂起的后台操作的请求;
重要方法:
1、CancelAsync 请求取消挂起的后台操作
2、RunWorkerAsync 开始执行后台操作
3、ReportProgress 引发ProgressChanged事件
重要事件:
1、DoWork 调用 RunWorkerAsync 时发生
2、ProgressChanged 调用 ReportProgress 时发生
3、RunWorkerCompleted 当后台操作已完成、被取消或引发异常时发生
另外还有三个重要的参数是RunWorkerCompletedEventArgs以及DoWorkEventArgs、ProgressChangedEventArgs。
BackgroundWorker的各属性、方法、事件的调用机制和顺序:
从上图可见在整个生活周期内发生了3次重要的参数传递过程:
参数传递1:此次的参数传递是将RunWorkerAsync(Object)中的Object传递到DoWork事件的DoWorkEventArgs.Argument,由于在这里只有一个参数可以传递,所以在实际应用往封装一个类,将整个实例化的类作为RunWorkerAsync的Object传递到DoWorkEventArgs.Argument;
参数传递2:此次是将程序运行进度传递给ProgressChanged事件,实际使用中往往使用给方法和事件更新进度条或者日志信息;
参数传递3:在DoWork事件结束之前,将后台线程产生的结果数据赋给DoWorkEventArgs.Result一边在RunWorkerCompleted事件中调用RunWorkerCompletedEventArgs.Result属性取得后台线程产生的结果。
另外从上图可以看到DoWork事件是在后台线程中运行的,所以在该事件中不能够操作用户界面的内容,如果需要更新用户界面,可以使用ProgressChanged事件及RunWorkCompleted事件来实现。
明白了BagkgroundWorker的事件调用顺序和参数传递机制之后在使用该组件用于多线程编程的时候就可以轻松许多了。
http://developer.51cto.com/art/200908/144374.htm
C#进度条实现实例是如何操作的呢?让我们看看下面的代码:
using System; using System.Collections.Generic; //C#进度条实现实例using System.ComponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; namespace 进度条2 //C#进度条实现实例{ public partial class Form1 : Form { private BackgroundWorker worker = new BackgroundWorker(); int N = 0; public Form1() { InitializeComponent(); worker.WorkerReportsProgress = true; worker.WorkerSupportsCancellation = true; worker.DoWork += new DoWorkEventHandler(DoWork); worker.ProgressChanged += new ProgressChangedEventHandler(ProgessChanged); worker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(CompleteWork); } public void DoWork( object sender, DoWorkEventArgs e) { e.Result = ComputeFibonacci(worker, e); } public void ProgessChanged( object sender, ProgressChangedEventArgs e) { progressBarX1.Value = e.ProgressPercentage; int V =(int)( e.ProgressPercentage / N); progressBarX1.Text = Convert.ToString(V) + "%"; } //C#进度条实现实例public void CompleteWork( object sender, RunWorkerCompletedEventArgs e) { progressBarX1.Text = "处理完毕!"; } private int ComputeFibonacci( object sender, DoWorkEventArgs e) { for (int i = 0; i <= 92800; i++) { if (worker.CancellationPending) { e.Cancel = true; return -1; } else { //C#进度条实现实例 worker.ReportProgress(i); //引发ProgessChanged事件 } } return -1; } private void btnStart_Click( object sender, EventArgs e) { N = 92800 / 100; progressBarX1.Maximum = 92800; worker.RunWorkerAsync(); //开始执行后台操作 } private void btnPause_Click( object sender, EventArgs e) { worker.CancelAsync(); //请求暂停后台操作 } } //C#进度条实现实例 }
winform - BackgroundWorker