在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: }
当该窗体被显示出来后,点击按钮让注册到该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: }