BackgroundWorker在单独的线程上执行操作

直接使用多线程有时候会带来莫名其妙的错误,不定时的发生,有时候会让程序直接崩溃,其实BackgroundWorker 类允许您在单独的专用线程上运行操作。可以通过编程方式创建 BackgroundWorker,也可以将它从“工具箱”的“组件”选项卡中拖到窗体上。如果在 Windows 窗体设计器中创建 BackgroundWorker,则它会出现在组件栏中,而且它的属性会显示在“属性”窗口中。

 

若要设置后台操作,请为 DoWork 事件添加一个事件处理程序。在此事件处理程序中调用耗时的操作。若要启动该操作,请调用 RunWorkerAsync。若要收到进度更新通知,请对 ProgressChanged 事件进行处理。若要在操作完成时收到通知,请对 RunWorkerCompleted 事件进行处理。

 

您必须非常小心,确保在 DoWork 事件处理程序中不操作任何用户界面对象。而应该通过 ProgressChanged 和 RunWorkerCompleted 事件与用户界面进行通信。BackgroundWorker 事件不跨 AppDomain 边界进行封送处理。请不要使用 BackgroundWorker 组件在多个 AppDomain 中执行多线程操作。

 

下面使用BackgroundWorker 完成斐波那契数列的后台运算操作,斐波那契数列:1,1,2,3,5,8...n=(n-1)+(n-2)

 

        BackgroundWorker bw;
        
// 要计算的斐波那契数列的位数
         int  numberToComputer  =   0 ;
        
// 当前完成的比例,按时间来算是不准确的
         int  hightest  =   0 ;
        
private   void  Form1_Load( object  sender, EventArgs e)
        {
            bw 
=   new  BackgroundWorker();
            
// 是否要报告进度
            bw.WorkerReportsProgress  =   true ;
            
// 是否允许取消
            bw.WorkerSupportsCancellation  =   true ;
            
// 工作事件
            bw.DoWork  +=   new  DoWorkEventHandler(bw_DoWork);
            
// 进度变化事件
            bw.ProgressChanged  +=   new  ProgressChangedEventHandler(bw_ProgressChanged);
            
// 运行完成事件
            bw.RunWorkerCompleted  +=   new  RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
        }

        
void  bw_RunWorkerCompleted( object  sender, RunWorkerCompletedEventArgs e)
        {
            
// 如果操作没有被取消,显示结果
             if  (e.Cancelled)
            {
                label1.Text 
=   " Cancelled " ;
            }
            
else
            {
                label1.Text 
=  e.Result.ToString();
            }
        }

        
void  bw_DoWork( object  sender, DoWorkEventArgs e)
        {
            
// 此处要注意不要跨线程操作
            
// 如果后台操作需要参数,请在调用 RunWorkerAsync 时给出参数。
            
// 在 DoWork 事件处理程序内部,可以从 
            
// oWorkEventArgs.Argument 属性中提取该参数。
            e.Result  =  Fib(( int )e.Argument, bw, e);
        }

        
void  bw_ProgressChanged( object  sender, ProgressChangedEventArgs e)
        {
            
// 显示当前进度
             this .progressBar1.Value  =  e.ProgressPercentage;
            lblPrecent.Text 
=  e.ProgressPercentage.ToString();
        }

        
///   <summary>
        
///  使用递归计算斐波那契数列的指定位数的值
        
///  1,1,2,3,5,8,13
        
///   </summary>
        
///   <param name="n"> 要计算的数 </param>
        
///   <param name="bwr"> 运行的后台进程 </param>
        
///   <param name="e"> 参数 </param>
        
///   <returns> 运算结果 </returns>
         long  Fib( int  n, BackgroundWorker bwr, DoWorkEventArgs e)
        {
            
long  result  =   0 ;
            
// 如果已被取消,设置参数值取消为真
             if  (bwr.CancellationPending)
            {
                e.Cancel 
=   true ;
            }
            
else
            {
                
if  (n  <   2 )
                {
                    result 
=   1 ;
                }
                
else
                {
                    result 
=  Fib(n  -   1 , bwr, e)  +  Fib(n  -   2 , bwr, e);
                }
                
// 计算进度
                 int  p  =  ( int )(( float )n  *   100   /  ( float )numberToComputer);
                
// 进度比上次有变化
                 if  (p  >  hightest)
                {
                    
// 最高进度为当前进度
                    hightest  =  p;
                    
// 报告进度
                    bw.ReportProgress(p);
                }
            }
            
return  result;
        }

        
private   void  btnStart_Click( object  sender, EventArgs e)
        {
            
// 如果不忙
             if  ( ! bw.IsBusy)
            {
                
// 初始进度条
                progressBar1.Value  =   0 ;
                
// 开始时最高进度为0
                hightest  =   0 ;
                
// 要进行耗时计算的数字
                numberToComputer  =  Convert.ToInt32(numericUpDown1.Value);
                
// 异步运行
                bw.RunWorkerAsync(numberToComputer);
            }
            
else
            {
                MessageBox.Show(
" busy " );
            }
        }

        
private   void  btnCancel_Click( object  sender, EventArgs e)
        {
            
// 取消正在进行的操作
            bw.CancelAsync();
        }

 耗时的操作(如下载和数据库事务)在长时间运行时可能会导致用户界面 (UI) 似乎处于停止响应状态。如果您需要能进行响应的用户界面,而且面临与这类操作相关的长时间延迟,则可以使用 BackgroundWorker 类方便地解决问题。

注:文章参考了MSDN许多

转载于:https://www.cnblogs.com/best/archive/2011/07/30/2122084.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值