基于事件的异步设计模式
时间:2011-01-09 来源:luoht
简单的BackgroundWorker使用:
private void backgroundWorker1_DoWork(System.Object sender, System.ComponentModel.DoWorkEventArgs e) { } private void backgroundWorker1_RunWorkerCompleted(object sender, System.ComponentModel.RunWorkerCompletedEventArgs e) { } private void backgroundWorker1_ProgressChanged(object sender, System.ComponentModel.ProgressChangedEventArgs e) { operationToolStripProgressBar.Value = e.ProgressPercentage; } private void btnCancel_Click(System.Object sender, System.EventArgs e) { backgroundWorker1.CancelAsync(); } private void btnStart_Click(System.Object sender, System.EventArgs e) { backgroundWorker1.RunWorkerAsync(object argument); }
基于事件的异步模式具有多线程应用程序的优点,同时隐匿了多线程设计中固有的许多复杂问题,例如:
1.“在后台”执行耗时任务(例如下载和数据库操作),但不会中断你的应用程序。
2.同时执行多个操作,每个操作完成时都会接到通知。
3.等待资源变得可用,但不会停止(“挂起”)你的应用程序。
使用事件和委托模型与挂起的异步操作通信
支持基于事件的异步模式的类将有一个或多个名为 MethodNameAsync 的方法。这些方法可能会创建同步版本的镜像,这些同步版本会在当前线程上执行相同的操作。此类还可能有一个 MethodNameCompleted 事件,而且它可能会有一个 MethodNameAsyncCancel(或只是 CancelAsync)方法。
例如PictureBox控件是一个支持基于事件的异步模式的典型组件。通过调用其 Load 方法来同步下载图像,但是如果图像很大,或者网络连接很慢,你的应用程序将停止(“挂起”),直到下载操作完成并且对 Load 的调用返回后才会继续执行。
如果你希望应用程序在加载图像时保持运行,那么可以调用 LoadAsync 方法,处理 LoadCompleted 事件。调用 LoadAsync 方法时,你的应用程序将继续运行,不受影响,而下载操作将在另一个线程上(“在后台”)继续。图像加载操作完成时,将会调用事件处理程序,事件处理程序可以检查 AsyncCompletedEventArgs 参数以确定下载是否已成功完成。
基于事件的异步模式要求异步操作可以取消,PictureBox 控件使用其 CancelAsync 方法来支持此要求。调用 CancelAsync 会提交一个停止挂起的下载的请求,任务取消时会引发 LoadCompleted 事件。
基于事件的异步模式一般设计
基于事件的异步模式可以采用多种形式,具体取决于某个特定类支持的操作的复杂程度。最简单的类可能只有一个 MethodNameAsync 方法和一个对应的 MethodNameCompleted 事件。更复杂的类可能有若干个 MethodNameAsync 方法(每种方法都有一个对应的 MethodNameCompleted 事件),以及这些方法的同步版本。这些类分别支持各种异步方法的取消、进度报告和增量结果。
异步方法可能还支持多个挂起的调用(多个并行调用),允许在此方法完成其他挂起的操作之前调用此方法任意多次。若要正确处理此种情况,必须让应用程序能够跟踪各个操作的完成。
基于事件的异步模式类设计:
public class AsyncExample { // Synchronous methods. public int Method1(string param); public void Method2(double param); // Asynchronous methods. public void Method1Async(string param); public void Method1Async(string param, object userState); public event Method1CompletedEventHandler Method1Completed; public void Method2Async(double param); public void Method2Async(double param, object userState); public event Method2CompletedEventHandler Method2Completed; public void CancelAsync(object userState); public bool IsBusy { get; } // Class implementation not shown. }
这里虚构的 AsyncExample 类有两个方法,都支持同步和异步调用。同步重载的行为类似于方法调用,它们对调用线程执行操作;如果操作很耗时,则调用的返回可能会有明显的延迟。异步重载将在另一个线程上启动操作,然后立即返回,允许在调用线程继续执行的同时让操作“在后台”执行。
异步方法重载
异步操作可以有两个重载:单调用和多调用。可以通过方法签名来区分这两种形式:多调用形式有一个额外的参数,即 userState。使用这种形式,在代码中可以多次调用 Method1Async(string param, object userState),而不必等待任何挂起的异步操作的完成。另一方面,如果尝试在前一个调用尚未完成时调用 Method1Async(string param),该方法将引发 InvalidOperationException。
多调用重载的 userState 参数可帮助您区分各个异步操作。应分别为各个 Method1Async(string param, object userState) 调用提供一个唯一值(例如 GUID 或哈希代码);这样,当各个操作完成时,事件处理程序便可以确定哪个操作的实例引发了完成事件。
跟踪挂起的操作
如果使用多调用重载,在代码中将需要跟踪挂起的任务的 userState 对象(任务 ID)。对于每个 Method1Async(string param, object userState) 调用,通常应生成一个新的、唯一的 userState 对象并将此对象添加到集合中。当对应于此 userState 对象的任务引发完成事件时,您的完成方法实现将检查 System.ComponentModel.AsyncCompletedEventArgs.UserState 并将此对象从集合中删除。在以这种方式使用时,userState 参数充当任务 ID 的角色。
另外,在对多调用重载的调用中的 userState 提供唯一值时,一定要小心。如果任务 ID 不唯一,将导致异步类引发 ArgumentException。
取消挂起的操作
我们必须能够在异步操作完成之前随时取消它们,这一点很重要。实现基于事件的异步模式的类将有一个 CancelAsync 方法(如果有多个异步方法)或 MethodNameAsyncCancel 方法(如果只有一个异步方法)。
允许多个调用的方法采用 userState 参数,此参数可用来跟踪各个任务的生存期。CancelAsync 采用 userState 参数,此参数允许取消特定的挂起任务。
一次只支持一个挂起的操作的方法(如 Method1Async(string param))是不可取消的。
接收进度更新和增量结果
符合基于事件的异步模式的类可以为跟踪进度和增量结果提供事件。此事件通常叫做 ProgressChanged 或 MethodNameProgressChanged,它对应的事件处理程序会带有一个 ProgressChangedEventArgs 参数。
ProgressChanged 事件的事件处理程序可以检查 System.ComponentModel.ProgressChangedEventArgs.ProgressPercentage 属性来确定异步任务完成的百分比。此属性的范围是 0 到 100,可用来更新 ProgressBar 的 Value 属性。如果有多个异步操作挂起,可以使用 System.ComponentModel.ProgressChangedEventArgs.UserState 属性来分辨出哪个操作在报告进度。
一些类可能会在异步操作继续时报告增量结果。这些结果将保存的派生自 ProgressChangedEventArgs 的类中,并显示为此派生类中的属性。可以在 ProgressChanged 事件的事件处理程序中访问这些结果,就像访问 ProgressPercentage 属性一样。如果有多个异步操作挂起,可以使用 UserState 属性来分辨出哪个操作在报告增量结果。
demo 下载