zoukankan      html  css  js  c++  java
  • 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看下运行结果:

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

  • 相关阅读:
    R中seurat等问题学习
    主成分分析PCA学习一条龙
    PAT 1116 Come on! Let's C [简单]
    PAT 1077 Kuchiguse [一般]
    R中的一些基础1106
    PAT 1036 Boys vs Girls[简单]
    稳定婚姻匹配问题模板
    University Entrace Examination zoj1023
    Stable Match
    7-1 玩转二叉树
  • 原文地址:https://www.cnblogs.com/leiOOlei/p/1952011.html
Copyright © 2011-2022 走看看