如果在BackgroundWorker运行过程中关闭窗体…

在Windows Form应用中,BackgroundWorker 类允许您在单独的专用线程上运行操作。耗时的操作(如下载和数据库事务)在长时间运行时可能会导致用户界面 (UI) 似乎处于停止响应状态。如果您需要能进行响应的用户界面,而且面临与这类操作相关的长时间延迟,则可以使用 BackgroundWorker 类方便地解决问题。由于操作是异步执行的,用户可能在异步操作执行过程中关闭当前窗体,而窗体的关闭会伴随着Dispose方法的执行。如果我们注册了BackgroundWorker的RunWorkerCompleted事件,并且在该事件处理程序中需要操作这个被Disposed的窗体,就会出现一些无法预知的异常。

一、一个简单的例子

我们写一个简单的例子来说明上述的场景:在一个非主窗体(主窗体的关闭会导致程序的终止)的Windows窗体中,一个BackgroundWorker被用于异步地执行一段耗时的操作。在我们的例子中,通过让线程休眠10秒来模拟这个“耗时操作”。方法backgroundWorker_RunWorkerCompleted是BackgroundWorker的RunWorkerCompleted事件处理方法,在这里我们通过MessageBox来显示当前窗体的IsDisposed属性值。注册到该BackgroundWorker的异步操作通过点击某个按钮开启。相关的代码如下所示。

   1: public partial class BgwForm : Form
   2: {
   3:     public BgwForm()
   4:     {
   5:         InitializeComponent();
   6:     }
   7:  
   8:     private void backgroundWorker_DoWork(object sender, DoWorkEventArgs e)
   9:     {
  10:         Thread.Sleep(10000);
  11:     }
  12:  
  13:     private void btnStart_Click(object sender, EventArgs e)
  14:     {
  15:         this.backgroundWorker.RunWorkerAsync();
  16:     }
  17:  
  18:     private void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
  19:     {
  20:         MessageBox.Show(string.Format("Is the form disposed? {0}", this.IsDisposed));
  21:     }
  22: }

image 当该窗体被显示出来后,点击按钮让注册到该BackgroundWorker的异步操作开始执行,然后在操作结束之前(10秒)将当前窗体关闭。10秒钟之后,如右图所示的一个MessageBox会显示出来,表明在执行BackgroundWorker的RunWorkerCompleted事件处理程序的时候,承载它的窗体已经被Dispose。如果在你将一些针对窗体的操作注册到RunWorkerCompleted事件上,操作一个Disposed窗体,很难保证能否正常进行。而实际上,我们通常注册该事件在窗体上进行一些状态信息的显示,既然窗体都被关闭,这些操作就无需执行。那么,有什么方式可以确保在窗体关闭的状态下阻止RunWorkerCompleted事件处理程序的执行呢?

二、通过IsDisposed属性判断窗体的状态

我们最容易想到的肯定是在RunWorkerCompleted事件处理程序通过窗体的IsDisposed属性判断窗体的状态,并根据状态进行相应的操作。

   1: private void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
   2: {
   3:     if (!this.IsDisposed)
   4:     { 
   5:         //Do Something
   6:     }
   7: }

三、在Closed事件中移除对RunWorkerCompleted事件的注册

我们还有另一种方案,那就是在关闭窗体的时候将对RunWorkerCompleted事件的注册接触掉,我们只需要将该操作注册到窗体的FormClosed事件中。

   1: private void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
   2: {
   3:     //Do Something
   4: }
   5:  
   6: private void BgwForm_FormClosed(object sender, FormClosedEventArgs e)
   7: {
   8:     this.backgroundWorker.RunWorkerCompleted -= backgroundWorker_RunWorkerCompleted;
   9: }
posted @ 2010-11-25 13:34  Artech  阅读(7820)  评论(36编辑  收藏  举报