【异步编程】异步模型 (APM)

使用 IASyncResult 设计模式的异步操作是通过名为 BeginOperationName 和 EndOperationName 的两个方法来实现的,这两个方法分别开始结束异步操作 OperationName。在调用 BeginOperationName 后,应用程序可以继续在调用线程上执行,同时异步操作在另一个线程上执行。 应用程序还应通过调用 EndOperationName 来获取异步操作的结果

1. 开始异步操作

BeginOperationName 方法开始异步操作 OperationName,并返回实现 IASyncResult 接口的对象。IASyncResult 对象存储有关异步操作的信息。 

异步操作的信息
成员类型说明
AsyncStateobject一个特定于应用程序的可选对象,其中包含有关异步操作的信息。
AsyncWaitHandleSystem.Threading.WaitHandle一个 WaitHandle可用来异步操作完成之前阻止应用程序执行。
CompletedSynchronouslybool一个值,指示异步操作是否是在用于调用 BeginOperationName 的线程上完成,而不是在单独的 ThreadPool 线程上完成。
IsCompletedbool一个值,指示异步操作是否已完成。

BeginOperationName 方法采用该方法的同步版本的签名中声明的任何参数(由值传递或由引用传递)。 BeginOperationName 方法签名中不包含任何输出参数。 BeginOperationName 方法签名另外还包括两个其他参数。

  • 第一个参数定义一个 AsyncCallback 委托,此委托引用在异步操作完成时调用的方法。 如果调用方不希望在操作完成后调用方法,它可以指定 null
  • 第二个参数是一个用户定义的对象。 此对象可用来异步操作完成调用的方法传递应用程序特定的状态信息。 如果 BeginOperationName 方法还采用其他一些操作特定的参数(例如,一个用于存储从文件读取的字节的字节数组),则 AsyncCallback 和应用程序状态对象将是 BeginOperationName 方法签名中的最后两个参数。

BeginOperationName 立即返回对调用线程的控制。 如果 BeginOperationName 方法引发异常,则会在开始异步操作之前引发异常。 如果 BeginOperationName 方法引发异常,则意味着没有调用回调方法

2. 结束异步操作

EndOperationName 方法用于结束异步操作 OperationName。 EndOperationName 方法的返回值与其同步对应方法的返回值类型相同,并且是特定于异步操作的。 例如,EndRead 方法返回从 FileStream 读取的字节数, EndGetHostByName 方法返回包含有关主机的信息的 IPHostEntry 对象。EndOperationName 方法采用该方法同步版本的签名中声明的所有输出参数或引用参数。 除了来自同步方法的参数外,EndOperationName 方法还包括 IASyncResult 参数。 调用方必须将对应调用返回的实例传递给 BeginOperationName

如果调用 EndOperationName 时 IASyncResult 对象表示的异步操作尚未完成,则 EndOperationName 将在异步操作完成之前阻止调用线程。 异步操作引发的异常是从 EndOperationName 方法引发的。 

  • 未定义多次使用同一 IASyncResult 调用 EndOperationName 方法的效果
  • 未定义Begin 方法未返回时,使用 IASyncResult 调用 EndOperationName 方法的效果

对于这两种未定义的情况,实施者应考虑引发 InvalidOperationException。此设计模式的实施者应通知调用方异步操作已通过以下步骤完成:将 IsCompeleted 设置为 true,调用异步回调方法(如果已指定一个回调方法),然后发送 AsyncWaitHandle信号。

3 访问异步执行的结果

正确的选择取决于应用程序是否有可以在操作完成时执行的指令。 如果应用程序在接收到异步操作结果之前不能进行任何其他工作,则必须在获得这些结果之前先阻止该应用程序进行其他工作。 若要在异步操作完成之前阻止应用程序,可以使用下列方法之一:

  • 应用程序的主线程调用 EndOperationName,阻止应用程序执行,直到操作完成之后再继续执行。

  • 使用 AsyncWaitHandle 来阻止应用程序执行,直到一个或多个操作完成。

在异步操作完成时不需要阻止的应用程序可使用下列方法之一:

  • 按以下方式轮询操作完成状态:定期检查 IsCompleted 属性,操作完成后调用 EndOperationName

  • 使用 AsyncCallback 委托来指定要在操作完成时调用的方法。 

4. 应用程序需要等待异步执行的结果,才能继续执行其他工作

如果应用无法在等待异步操作结果期间继续执行其他工作,必须阻止应用一直到操作完成。 可以使用以下两种方式在应用等待异步操作完成期间阻止应用的主线程: 

  • 使用异步操作的 BeginOperationName 方法返回的 IAsyncResult 的 AsyncWaitHandle 属性。

  • 在应用程序的主线程上调用EndOperationName

4.1 使用 AsyncWaitHandle 阻止应用程序的执行

在异步操作完成前使用一个或多个 WaitHandle 对象阻止的应用。使用 AsyncWaitHandle 调用 WaitOne 方法,对单个操作阻止应用。 若要在等待一组异步操作完成期间阻止应用,请将相关的 AsyncWaitHandle 存储到数组,调用 System.Threading.WaitHandle.WaitAll 方法。若要在等待一组异步操作中任一操作完成期间阻止应用,请将相关的 AsyncWaitHandle 对象存储到数组中,并调用 System.Threading.WaitHandle.WaitAny方法。

以下代码示例展示了如何使用 DNS 类中的异步方法,检索用户指定计算机的域名系统信息。 

using System;
using System.Net;
using System.Net.Sockets;
using System.Threading;

namespace AsynchronousOperations
{
    public class WaitUntilOperationCompletes
    {
        public static void Main()
        {
            string strHostNameOrAddr = "127.0.0.1";
            IAsyncResult result = Dns.BeginGetHostEntry(strHostNameOrAddr, null, null);

            result.AsyncWaitHandle.WaitOne();

            try 
            {
                IPHostEntry host = Dns.EndGetHostEntry(result);
                string[] aliases = host.Aliases;
                IPAddress[] addresses = host.AddressList;
				
                if (aliases.Length > 0)
                {
                    Console.WriteLine("Aliases");
                    for (int i = 0; i < aliases.Length; i++)
                    {
                        Console.WriteLine("{0}", aliases[i]);
                    }
                }
				
                if (addresses.Length > 0)
                {
                    Console.WriteLine("Addresses");
                    for (int i = 0; i < addresses.Length; i++)
                    {
                        Console.WriteLine("{0}",addresses[i].ToString());
                    }
                }
            }
            catch (SocketException e)
            {
                Console.WriteLine("Exception occurred while processing the request: {0}",  e.Message);
            }
        }
    }
}

4.2 通过结束异步操作来阻止应用程序执行

下面的代码示例展示了如何使用 Dns 类中的异步方法,检索用户指定计算机的域名系统信息。


using System;
using System.Net;
using System.Net.Sockets;

namespace AsynchronousOperations
{
    public class BlockUntilOperationCompletes
    {
        public static void Main(string[] args)
        {
            string strHostNameOrAddr = "127.0.0.1";
            IAsyncResult result = Dns.BeginGetHostEntry(strHostNameOrAddr, null, null);

            try 
            {
                IPHostEntry host = Dns.EndGetHostEntry(result);
                string[] aliases = host.Aliases;
                IPAddress[] addresses = host.AddressList;
				
                if (aliases.Length > 0)
                {
                    Console.WriteLine("Aliases");
                    for (int i = 0; i < aliases.Length; i++)
                    {
                        Console.WriteLine("{0}", aliases[i]);
                    }
                }
				
                if (addresses.Length > 0)
                {
                    Console.WriteLine("Addresses");
                    for (int i = 0; i < addresses.Length; i++)
                    {
                        Console.WriteLine("{0}",addresses[i].ToString());
                    }
                }
            }
            catch (SocketException e)
            {
                Console.WriteLine("An exception occurred while processing the request: {0}", e.Message);
            }
        }
    }
}

 

5. 应用程序在等待异步执行的结果时,可以同步完成其他工作

如果应用可以在等待异步操作结果期间继续执行其他工作,不得阻止应用一直到操作完成。 请使用下列方法之一,在应用等待异步操作完成期间继续执行指令:

  • 使用异步操作的 BeginOperationName 方法返回的 IAsyncResult 的 IsCompleted 属性,确定操作是否已完成。

  • 使用 AsyncCallback 委托,在单独的线程中处理异步操作结果。

5.1 轮询异步操作的状态

下面的代码示例展示了如何使用 Dns 类中的异步方法,检索用户指定计算机的域名系统信息。 此示例启动异步操作,然后在控制台打印句点 ("."),直到操作完成。

using System;
using System.Net;
using System.Net.Sockets;
using System.Threading;

namespace AsynchronousOperations
{
    public class PollUntilOperationCompletes
    {
        static void UpdateUserInterface()
        {
            Console.Write(".");
        }
		
        public static void Main(string[] args)
        {
            string strHostNameOrAddr = "127.0.0.1";
            IAsyncResult result = Dns.BeginGetHostEntry(strHostNameOrAddr, null, null);

            while (result.IsCompleted != true)
            {
                UpdateUserInterface();
            }

            try 
            {
                IPHostEntry host = Dns.EndGetHostEntry(result);
                string[] aliases = host.Aliases;
                IPAddress[] addresses = host.AddressList;
				
                if (aliases.Length > 0)
                {
                    Console.WriteLine("Aliases");
                    for (int i = 0; i < aliases.Length; i++)
                    {
                        Console.WriteLine("{0}", aliases[i]);
                    }
                }
				
                if (addresses.Length > 0)
                {
                    Console.WriteLine("Addresses");
                    for (int i = 0; i < addresses.Length; i++)
                    {
                        Console.WriteLine("{0}",addresses[i].ToString());
                    }
                }
            }
            catch (SocketException e)
            {
                Console.WriteLine("An exception occurred while processing the request: {0}", e.Message);
            }
        }
    }
}

5.2 使用 AsyncCallback 委托结束异步操作

下面的代码示例展示了如何使用 Dns 类中的异步方法,检索用户指定计算机的域名系统 (DNS) 信息。

using System;
using System.Net;
using System.Net.Sockets;

namespace AsynchronousOperations
{
    public class UseDelegateForAsyncCallback
    {
        public static void Main()
        {
            string strHostNameOrAddr = "127.0.0.1";
            Dns.BeginGetHostEntry(strHostNameOrAddr, new AsyncCallback(ProcessDnsInformation), strHostNameOrAddr);
			
            for (int i = 0; i < 10; i++)
            {
                Console.WriteLine("...");
            }

            Console.ReadKey();
       }

        static void ProcessDnsInformation(IAsyncResult result)
        {
            string hostName = (string) result.AsyncState;
            try 
            {
                IPHostEntry host = Dns.EndGetHostEntry(result);
                string[] aliases = host.Aliases;
                IPAddress[] addresses = host.AddressList;
                if (aliases.Length > 0)
                {
                    Console.WriteLine("Aliases");
                    for (int j = 0; j < aliases.Length; j++)
                    {
                        Console.WriteLine("{0}", aliases[j]);
                    }
                }
                if (addresses.Length > 0)
                {
                    Console.WriteLine("Addresses");
                    for (int k = 0; k < addresses.Length; k++)
                    {
                        Console.WriteLine("{0}",addresses[k].ToString());
                    }
                }
            }
            catch (SocketException e)
            {
                Console.WriteLine("An exception occurred while processing the request: {0}", e.Message);
            }
        }
    }
}

5.3 使用 AsyncCallback 委托和状态对象

下面的代码示例展示了如何使用 Dns 类中的异步方法,检索用户指定计算机的域名系统 (DNS) 信息。 此示例定义并使用 HostRequest 类存储状态信息。 HostRequest 对象是针对用户输入的每个计算机名进行创建。 此对象传递给 BeginGetHostByName 方法。 每当请求完成时,都会调用 ProcessDnsInformation 方法。 HostRequest 对象是使用 AsyncState 属性进行检索。 ProcessDnsInformation 方法使用 HostRequest 对象,存储请求返回的 IPHostEntry 或请求抛出的 SocketException。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhy29563

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值