BackgroundWorker 类
在单独的线程上执行操作。
BackgroundWorker 类允许您在单独的专用线程上运行操作。耗时的操作(如下载和数据库事务)在长时间运行时可能会导致用户界面 (UI) 似乎处于停止响应状态。
如果您需要能进行响应的用户界面,而且面临与这类操作相关的长时间延迟,则可以使用 BackgroundWorker 类方便地解决问题。
若要在后台执行耗时的操作,请创建一个 BackgroundWorker,侦听那些报告操作进度并在操作完成时发出信号的事件。
可以通过编程方式创建 BackgroundWorker,也可以将它从“工具箱”的“组件”选项卡中拖到窗体上。
如果在 Windows 窗体设计器中创建 BackgroundWorker,则它会出现在组件栏中,而且它的属性会显示在“属性”窗口中。
若要设置后台操作,请为 DoWork 事件添加一个事件处理程序。在此事件处理程序中调用耗时的操作。
若要启动该操作,请调用 RunWorkerAsync。若要收到进度更新通知,请对 ProgressChanged 事件进行处理。
若要在操作完成时收到通知,请对 RunWorkerCompleted 事件进行处理。
说明:
您必须非常小心,确保在 DoWork 事件处理程序中不操作任何用户界面对象。
而应该通过 ProgressChanged 和 RunWorkerCompleted 事件与用户界面进行通信。
BackgroundWorker 事件不跨 AppDomain 边界进行封送处理。请不要使用 BackgroundWorker 组件在多个 AppDomain 中执行多线程操作。
如果后台操作需要参数,请在调用 RunWorkerAsync 时给出参数。在 DoWork 事件处理程序内部,可以从 DoWorkEventArgs.Argument 属性中提取该参数。
请注意:在你调用的后台线程里一定要加上Thread.Sleep(100)这一句,
别小看这条语句,他使得后台线程运行时可以释放出足够的时间在前台做操作。
实例如下,首先看界面

程序实现点击Start按钮在后台线程生成制定数量的GUID,写到listbox中,点击End后停止后台线程。
1.首先写好函数,此函数我们要在后台运行处理,同事报告进度,还要返回结果
private int GetDate(BackgroundWorker worker, DoWorkEventArgs e)
{
//获取传入的参数
int maxRecord = (int)e.Argument;
int percent = 0;
for (int i = 0; i < maxRecord; i++)
{
if (worker.CancellationPending)
return i;
percent = (int)((double)i / (double)maxRecord * 100);
//下边这句执行完后激活ProgressChanged事件
worker.ReportProgress(percent, new KeyValuePair<int, string>(i, Guid.NewGuid().ToString()));
//让后台线程Sleep一段时间非常有用,这样前台UI中用户可以做一些其他的输入操作,不至于傻等
Thread.Sleep(100);
}
return maxRecord;
}
2.设置backgroundWorker1的属性,WorkerReportProgress=true,WorkerSupportsCancellation=true
3.写DoWork,ProgressChanged,RunWorkerCompleted事件
private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
{
try
{
BackgroundWorker worker = sender as BackgroundWorker;
e.Result = GetDate(worker, e);
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
throw;
}
}
private void backgroundWorker1_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
KeyValuePair<int, string> record = (KeyValuePair<int, string>)e.UserState;
this.label1.Text = string.Format("已经计算了{0}条", record.Key);
this.progressBar1.Value = e.ProgressPercentage;
this.listBox1.Items.Add(record.Value);
}
private void backgroundWorker1_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
try
{
this.label1.Text = string.Format("已经完成,总计{0}", e.Result);
this.btn_start.Enabled = true;
btn_end.Enabled = false;
}
catch (Exception ex)
{
MessageBox.Show(ex.Message);
}
}
4.点击Start时的代码
private void btn_start_Click(object sender, EventArgs e)
{
if (this.backgroundWorker1.IsBusy)
return;
this.listBox1.Items.Clear();
this.backgroundWorker1.RunWorkerAsync(100);
this.btn_start.Enabled = false;
this.btn_end.Enabled = true;
}
点击Start后,this.backgroundWorker1.RunWorkerAsync(100),让Getate生成100个GUID,写入listBox中。
开始执行DoWork中的代码,同时将参数100封装在DoWorkEventArgs中,通过e.Argument获取,类型是object,需要转换程你需要的类型。
5.最后写取消事件
private void btn_end_Click(object sender, EventArgs e)
{
this.backgroundWorker1.CancelAsync();
}
OK看下运行结果:

运行时界面可以拖动,不是死锁状态。
浙公网安备 33010602011771号