【FastReport教程】介绍C#中的异步编程(下)
【下载FastReport.Net最新版本】
异步编程模型出现在.Net Framework的第一个版本中。APM允许使用两种方法创建同步方法的异步版本 - Begin 和End 。 所以,只有两种方法:
public IAsyncResult Begin{MethodName}(TIn[] args, AsyncCallback callback, object userState =null) { … }
和:
public TResult End{MethodName}(IAsyncResult result) { ... }
Begin {MethodName}方法启动异步操作。它接受参数args,callback--执行异步方法后调用的方法的委托,userState对象,用于将有关特定应用程序状态的信息传递给异步操作结束时调用的方法。 该方法返回IAsyncResult类型的对象,该对象存储有关异步操作的信息。 End {MethodName}方法终止异步操作。它接受一个输入对象,类型为IAsyncResult,并返回TResult,它实际上返回此方法的同步副本中定义的类型。 让我们看一下简化示例中如何使用此模板:
public void Button_Click (...) { WebRequest request = WebRequest.Create(url); request.BeginGetResponse(Callback , request); } public void Callback(IAsyncResult ar) { WebRequest request = (WebRequest) ar.AsyncState; try { var response = request.EndGetResponse(ar); // Code does something with successful response } catch (WebException e) { // Error handling code } }
我们在按钮单击事件处理程序中调用了Begin方法。作为参数,我们将回调传递给此方法。并且,已经在回调本身中我们称为对方法 - 结束。 异步编程模型的缺点包括:
- 创建回调函数的必要性;
- 缺乏中止操作的方法。
- 没有拒绝调用回调方法的方法,如果它是在操作开始时传递的。
- 没有通知操作进度或中期结果。
基于事件的异步模式
这种异步编程模式出现在.Net框架的第二个版本中。它基于事件和异步方法。实现此模板的类将包含方法MethodNameAsync和MethodNameAsyncCancel(如果需要操作取消处理)和MethodNameCompleted事件。在同一个类中,您可以放置??使用相同线程的方法的同步版本。大多数情况下,在使用Web服务时会使用此模板。例如,ajax实现基于事件的异步模式。您只能在MethodNameCompleted事件处理程序中获取异步操作的结果并处理错误。
基于事件的异步编程模式解决了前一个问题:
- 声明获取异步操作结果的方法;
- 没有通知操作进度的机制。
但是,此模板仍有许多缺点:
- 无法将操作调用的上下文(用户数据)传送到结果处理方法;
- 并非所有操作都可以中断。只支持队列中一个操作的方法不能被中断;
- 无法在上下文中指定将调用哪个线程回调方法。
基于任务的异步模式(TAP)
异步编程的第三个模板出现在.Net Framework 4.0中。从标题中可以清楚地看出它基于任务的使用。TAP的基础是System.Threading.Tasks.Task和System.Threading.Tasks.Task 两种类型。
TAP允许开发人员在单个方法中定义异步函数。现在不需要创建异步操作的开始和结束的函数,然后还有回调函数。这当然有利于程序员的工作,降低了进入技术的门槛,并简单地使编程变得愉快。 TAP使用任务来执行操作。对于每个任务,使用一个单独的线程,该线程取自线程池。任务完成后,线程返回池。
修饰符“async” - 此修饰符应用于方法或lambda表达式或匿名方法 - 它表示该方法是异步的,并表示此方法中出现一次或多次wait语句的可能性。 我们来看一个方法定义的例子:
public async Task<int>MyProcessAsync() { … Var Overtime = await new ERP().ProcessOvertime(emp); … }
注意关键字async和await。这些是表示正在使用基于任务的异步模式的运算符。async修饰符表示该方法是异步的。并且可以在方法内调用await语句一次或多次。它暂停执行任务,直到获得结果,同时线程继续工作。
这是一个从生活中使用TAP的例子。调用Web服务:
static async Task<string> SendMessageAsync() { var client = new MyServiceClient(); var task = Task.Factory.StartNew(() => client.SendMessageAsync("Message")); var result = await task; return result; }
这是另一种调用方式,更简单,更容易理解:
static async Task<string> SendMessageAsyncNew() { var client = new MyServiceClient(); var result = await client.SendMessageAsync("Message"); return result; }
使用await的这个“轻量级”版本可以在.Net Framework 4.5中找到。 基于异步任务的方法解决了以前模板的大多数问题。然后你和中断异步操作的能力,以及一个简单的实现方法,跟踪操作进度的能力。 目前,Microsoft建议在开发组件时使用此模板实现异步调用。 至于使用这个模板的权宜之计。使用TAP将增加服务器的带宽。但是,如果您有少量流量(例如,客户端 - 服务器),则创建异步流程的成本可能会抵消收益。在这种情况下,同步方法将更快地工作。