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看下运行结果:

运行时界面可以拖动,不是死锁状态。

posted on 2011-02-12 10:48  leiOOlei  阅读(492)  评论(0编辑  收藏  举报