异步编程概述
使用 IAsyncResult 设计模式的异步操作是通过名为 Begin操作名称 和 End操作名称 的两个方法来实现的,这两个方法分别开始和结束异步操作 操作名称。 例如,FileStream 类提供 BeginRead 和 EndRead 方法来从文件异步读取字节。 这两个方法实现了 Read 方法的异步版本。
在调用 Begin操作名称 后,应用程序可以继续在调用线程上执行指令,同时异步操作在另一个线程上执行。 每次调用 Begin操作名称 时,应用程序还应调用 End操作名称 来获取操作的结果。
开始异步操作
BeginOperationName 方法开始异步操作 OperationName,并返回实现 IAsyncResult 接口的对象。 IAsyncResult 对象存储有关异步操作的信息。 下表提供了有关异步操作的信息。
成员 |
说明 |
---|---|
一个可选的应用程序特定的对象,包含有关异步操作的信息。 |
|
一个 WaitHandle,可用来在异步操作完成之前阻止应用程序执行。 |
|
一个值,指示异步操作是否是在用于调用 Begin操作名称 的线程上完成,而不是在单独的 ThreadPool 线程上完成。 |
|
一个值,指示异步操作是否已完成。 |
Begin操作名称 方法带有该方法的同步版本的签名中声明的任何参数(由值传递或由引用传递)。 Begin操作名称 方法签名中不包含任何输出参数。 Begin操作名称 方法签名另外还包括两个参数。 在这两个参数中,第一个参数定义一个 AsyncCallback 委托,此委托引用在异步操作完成时调用的方法。 如果调用方不希望在操作完成后调用方法,它可以指定 null(在 Visual Basic 中为 Nothing)。 第二个参数是一个用户定义的对象。 此对象可用来向异步操作完成时调用的方法传递应用程序特定的状态信息。 如果 Begin操作名称 方法还带有其他一些操作特定的参数(例如,一个用于存储从文件读取的字节的字节数组),则 AsyncCallback 和应用程序状态对象将是 Begin操作名称 方法签名中的最后两个参数。
BeginOperationName 立即将控制权返回给调用线程。 如果 Begin操作名称 方法引发异常,则会在开始异步操作之前引发异常。 如果 Begin操作名称 方法引发异常,则意味着没有调用回调方法。
结束异步操作
End操作名称 方法可结束异步操作 操作名称。 End操作名称 方法的返回值与其同步副本的返回值类型相同,并且是特定于异步操作的。 例如,EndRead 方法返回从 FileStream 读取的字节数,EndGetHostByName 方法返回包含有关主机的信息的 IPHostEntry 对象。 End操作名称 方法带有该方法同步版本的签名中声明的所有输出参数或引用参数。 除了来自同步方法的参数外,End操作名称 方法还包括 IAsyncResult 参数。 调用方必须将对应调用返回的实例传递给 Begin操作名称。
如果调用 End操作名称 时 IAsyncResult 对象表示的异步操作尚未完成,则 End操作名称 将在异步操作完成之前阻止调用线程。 异步操作引发的异常是从 End操作名称 方法引发的。 未定义多次使用同一 IAsyncResult 调用 End操作名称 方法的效果。 同样,也未定义使用非相关的 Begin 方法返回的 IAsyncResult 调用 End操作名称 方法的效果。
注意 |
---|
对于这两种未定义的情况,实施者应考虑引发 InvalidOperationException。 |
注意 |
---|
此设计模式的实施者应通知调用方异步操作已通过以下步骤完成:将 IsCompleted 设置为 true,调用异步回调方法(如果已指定一个回调方法),然后发送 AsyncWaitHandle 信号。 |
对于访问异步操作的结果,应用程序开发人员有若干种设计选择。 正确的选择取决于应用程序是否有可以在操作完成时执行的指令。 如果应用程序在接收到异步操作结果之前不能进行任何其他工作,则必须先阻止该应用程序进行其他工作,等到获得这些操作结果后再继续进行。 若要在异步操作完成之前阻止应用程序,您可以使用下列方法之一:
从应用程序的主线程调用 End操作名称,阻止应用程序执行,直到操作完成之后再继续执行。 有关演示此方法的示例,请参见 通过结束异步操作来阻止应用程序执行。
使用 AsyncWaitHandle 来阻止应用程序执行,直到一个或多个操作完成之后再继续执行。 有关演示此方法的示例,请参见 使用 AsyncWaitHandle 阻止应用程序的执行。
在异步操作完成时不需要阻止的应用程序可使用下列方法之一:
按以下方式轮询操作完成状态:定期检查 IsCompleted 属性,操作完成后调用 End操作名称。 有关演示此方法的示例,请参见 轮询异步操作的状态。
使用 AsyncCallback 委托来指定操作完成时要调用的方法。 有关演示此方法的示例,请参见 使用 AsyncCallback 委托结束异步操作。