cpu密集型运算,如果直接在 UI 线程执行这样的运算就会出现UI不响应的问题。解决这类问题的主要途径是使用多线程,启动一个后台线程,把运算操作放在这个后台线程中完成。但是原生接口的线程操作有一些难度,如果要更进一步的去完成线程间的通信就会难上加难。
解决这类问题。虽然BackgroundWorker 类使用起来比较简单,但其中还是有一些需要注意的细节,下面我们就通过 demo 程序介绍它的主要用法。我们在 demo中计算1到100的累加和,为了演示,每次计算都 sleep 600毫秒,demo 的UI为:
用法概述
函数中添加耗时的运算,然后调用它的RunWorkerAsync方法就可以了。
显示在UI上该怎么办?
用户指定求和的范围呢!所以需要把100作为参数传递给计算过程。在概述中我们通过调用RunWorkerAsync方法启动计算过程,其实这个方法可以接受一个 object 类型的参数。通过它我们就可以把任何数据传递给计算过程:
函数通过参数 e 的Argument属性传来了我们期望的运算信息。
显示当前进度的同时,还希望能实时的把计算的中间结果显示在UI上。当然,BackgroundWorker对这个用例也提供了很好的支持。它允许我们在执行计算的过程中给UI线程发送消息,下面看看具体的做法:
属性设置为 true,然后为ProgressChanged 事件添加处理方法:
</span><span style="color: #008000;">//</span><span style="color: #008000;">如果有更多的信息需要传递,可以使用 e.UserState 传递一个<a href="/tag/zidingyi/" target="_blank" class="keywords">自定义</a>的类型。
</span><span style="color: #008000;">//</span><span style="color: #008000;">这是一个 object 类型的对象,您可以通过它传递任何类型。
</span><span style="color: #008000;">//</span><span style="color: #008000;">我们仅把当前 sum 的值通过 e.UserState 传回,并通过<a href="/tag/xianshi/" target="_blank" class="keywords">显示</a>在窗口上。</span>
<span style="color: #0000ff;">string</span> message =<span style="color: #000000;"> e.UserState.ToString();
</span><span style="color: #0000ff;">this</span>.labelSum.Text =<span style="color: #000000;"> message;
}
方法:
</span><span style="color: #0000ff;">int</span> sum = <span style="color: #800080;">0</span><span style="color: #000000;">;
</span><span style="color: #0000ff;">for</span> (<span style="color: #0000ff;">int</span> i = <span style="color: #800080;">0</span>; i <= endNumber; i++<span style="color: #000000;">)
{
sum </span>+=<span style="color: #000000;"> i;
</span><span style="color: #0000ff;">string</span> message = <span style="color: #800000;">"</span><span style="color: #800000;">Current sum is: </span><span style="color: #800000;">"</span> +<span style="color: #000000;"> sum.ToString();
</span><span style="color: #008000;">//</span><span style="color: #008000;">ReportProgress <a href="/tag/fangfa/" target="_blank" class="keywords">方法</a>把信息传递给 ProcessChanged 事件处理<a href="/tag/hanshu/" target="_blank" class="keywords">函数</a>。
</span><span style="color: #008000;">//</span><span style="color: #008000;">第一个参数类型为 int,表示执行进度。
</span><span style="color: #008000;">//</span><span style="color: #008000;">如果有更多的信息需要传递,可以使用 ReportProgress 的第二个参数。
</span><span style="color: #008000;">//</span><span style="color: #008000;">这里我们给第二个参数传进去一条消息。</span>
<span style="color: #000000;"> bgWorker.ReportProgress(i,message);
Thread.Sleep(
}
}
用户取消当前的操作是一个基本的设计,BackgroundWorker自然有很好的支持:
属性一样,如果要支持取消操作我们需要设置 WorkerSupportsCancellation属性为 true。并且还要在BGWorker_DoWork方法中进行支持,在 for 循环中 Thread.Sleep(600)后面添加代码:
<span style="color: #0000ff;">if (bgWorker.CancellationPending == <span style="color: #0000ff;">true<span style="color: #000000;">)
{
e.Cancel = <span style="color: #0000ff;">true<span style="color: #000000;">;
<span style="color: #0000ff;">break<span style="color: #000000;">;
}
用户点击的取消按钮,就退出当前的计算过程。下面是点击取消按钮时要调用的代码:
支持取消操作了,赶快试试吧!
</span><span style="color: #008000;">//</span><span style="color: #008000;">计算已经结束,需要禁用取消按钮。</span>
<span style="color: #0000ff;">this</span>.btnCancel.Enabled = <span style="color: #0000ff;">false</span><span style="color: #000000;">;
</span><span style="color: #008000;">//</span><span style="color: #008000;">计算过程中的异常会被抓住,在这里可以进行处理。</span>
<span style="color: #0000ff;">if</span> (e.Error != <span style="color: #0000ff;">null</span><span style="color: #000000;">)
{
Type errorType </span>=<span style="color: #000000;"> e.Error.GetType();
</span><span style="color: #0000ff;">switch</span><span style="color: #000000;"> (errorType.Name)
{
</span><span style="color: #0000ff;">case</span> <span style="color: #800000;">"</span><span style="color: #800000;">ArgumentNullException</span><span style="color: #800000;">"</span><span style="color: #000000;">:
</span><span style="color: #0000ff;">case</span> <span style="color: #800000;">"</span><span style="color: #800000;">MyException</span><span style="color: #800000;">"</span><span style="color: #000000;">:
</span><span style="color: #008000;">//</span><span style="color: #008000;">do something.</span>
<span style="color: #0000ff;">break</span><span style="color: #000000;">;
</span><span style="color: #0000ff;">default</span><span style="color: #000000;">:
</span><span style="color: #008000;">//</span><span style="color: #008000;">do something.</span>
<span style="color: #0000ff;">break</span><span style="color: #000000;">;
}
}
</span><span style="color: #008000;">//</span><span style="color: #008000;">计算结果信息:e.Result
</span><span style="color: #008000;">//</span><span style="color: #008000;">use it do something.</span>
}
函数会在DoWork 事件处理函数返回后被调用。通过它我们可以进行一些运算结束后的操作,比如禁用取消按钮,异常处理,结果显示等。方法中设置 e.Result属性,如:
功能完善且使用简便,实在是处理异步耗时操作的利器!