当添加 web 引用时,visual studio 生成既可以同步也可以异步调用 xml web services 的代理类。通过使用异步方法,您可以在等待 xml web services 响应时继续使用调用线程。这使您有效地使用客户端应用程序中现有的线程集,如果客户端是 web 应用程序,这可能十分关键。xml web services 不需要特殊的配置来支持异步调用,它也不知道客户端以何种模式调用它,这是因为对于同步调用和异步调用来说,请求和响应消息行为是相同的。客户端异步调用只是使用系统 i/o 完成端口有效地等待服务器的响应。
对于每个同步方法,生成的代理类还包含相应的 begin 和 end 方法。因此,如果 xml web services 方法的名称是 converttemperature,代理类中同步方法的名称也是 converttemperature,支持异步调用的其他代理方法是 beginconverttemperature 和 endconverttemperature。
异步调用 xml web services 这一操作分为两个步骤。第一步(调用 begin 方法)启动 xml web services 调用。第二步(调用 end 方法)完成 xml web services 调用并返回 xml web services 响应。
begin 方法返回 system.web.services.protocols.webclientasyncresult 对象。该对象(它实现 system.iasyncresult)达到两个目的。首先,它提供有关挂起异步调用的状态。例如,iscompleted 属性指示操作是否已完成。其次,通过将该对象传递给 end 方法,代理可以识别将完成哪个请求。这非常重要,因为您可以同时进行多个异步调用。
有多种方法确定异步 xml web services 调用何时完成:
- 第一种方法是向 begin 方法提供回调委托。在 xml web services 返回其响应之后,线程将从线程池调用回调;这很可能是一个与用来调用 begin 方法的线程不同的线程。通常,将在该回调中调用 end 方法。因为回调功能在等待响应时不阻塞线程,所以使用回调是管理线程使用的最有效的方法。
- 第二种方法是使用 iasyncresult.asyncwaithandle 对象的一种 waithandle 方法进行等待。当使用 waithandle 类的方法时,客户端还可以指定超时,在超时之后,它将放弃等待从它调用的 xml web services 返回结果。
- 第三种方法是轮询 iasyncresult.iscompleted 的值。当该属性返回 true 时,xml web services 响应可用。
- 第四种方法是仅仅直接调用 end 方法。因为该方法使用 iasyncresult.asyncwaithandle,所以在异步操作完成之前,该方法将不返回。
waithandle 类使客户端可以异步调用并等待以下对象:
- 单个 xml web services (waithandle.waitone),
- 多个 xml web services 中的第一个 (waithandle.waitany),或
- 多个 xml web services 中的所有服务 (waithandle.waitall) 以返回结果。
如果同时进行多个异步 xml web services 调用,那么您可能要使用 waithandle.waitall 方法来阻塞线程,直到所有调用完成为止。如果要在结果到达时对其进行处理,可以使用 waithandle.waitany 方法。该方法将指示一个操作已经完成并将识别该已完成的操作。
当从 web 应用程序异步访问 xml web services 时,使用 waithandle 类的方法是首选方法。否则,如果 web 应用程序中的执行在异步进程调用回调函数之前完成(如上一节所述),那么当原始函数的完成导致 web 窗体的实例毁坏时,它可能永远不会有机会完成此操作。
如果要中止异步 xml web services 调用,请将从 begin 方法返回的 iasyncresult 转换为 webclientasyncresult 并调用其 abort 方法。
有关更多信息,请参见asp”>与 xml web services 进行异步通信。
使用托管代码中的回调函数异步调用 xml web services
- 创建要从其中访问 xml web services 的应用程序。该应用程序甚至可以是另一个 xml web services。
- 为应用程序将与之交互的 xml web services 添加 web 引用。有关说明,请参见asp”>添加和移除 web 引用。
- 实现回调函数。回调函数包含对 xml web services 方法的 end 方法的调用。
- 在要从其中访问 xml web services 的客户端代码中创建代理对象的实例。
- 调用 xml web services 方法的 begin 方法。
- 在客户端应用程序中继续进行任何其他的处理。
- xml web services 完成处理并将结果返回给代理后,代理将调用回调函数。
下面显示的代码来自一个 windows 应用程序,该应用程序正在访问一个 xml web services,而该应用程序具有一个对该 xml web services 的 web 引用 (translator),该引用包含代理类 (service1),而该代理类具有异步调用 xml web services 的 begin 方法 (beginlongprocess) 和完成调用并获得结果的 end 方法 (endlongprocess)。在此示例中,xml web services 接受字符串,执行某些耗时的转换,然后返回结果。在启动对 xml web services 方法的异步调用之后,事件处理程序 button1_click 将控制返回给客户端。当 xml web services 返回其结果时,它调用回调函数。
visual basic private sub button1_click(byval sender as system.object, _ byval e as system.eventargs) _ handles button1.click dim cservice as new translator.service1() cb is essentially a function pointer to servicecallback. dim cb as new asynccallback(addressof servicecallback) call the begin method of the proxy class to initiate the asynchronous call to the xml web service method. cservice.beginlongprocess(textbox1.text, cb, cservice) end sub public sub servicecallback(byval ar as iasyncresult) dim cservice as translator.service1 retrieve the original state for the proxy. cservice = ar.asyncstate retrieve results by calling the end method of the proxy class. textbox2.text = cservice.endlongprocess(ar) end sub // c# private void button1_click(object sender, system.eventargs e) { translator.service1 cservice = new translator.service1(); // cb is essentially a function pointer to servicecallback. asynccallback cb = new asynccallback(servicecallback); // call the begin method of the proxy class to initiate the // asynchronous call to the xml web service method. cservice.beginlongprocess(textbox1.text, cb, cservice); } public void servicecallback(iasyncresult ar) { // retrieve the original state for the proxy. translator.service1 cservice = (translator.service1)ar.asyncstate; // retrieve results by calling the end method of the proxy class textbox2.text = cservice.endlongprocess(ar); }
使用托管代码中的 waithandle 异步调用 xml web services
- 创建要从其中访问 xml web services 的应用程序。该应用程序甚至可以是另一个 xml web services。
- 为应用程序将与之交互的 xml web services 添加 web 引用。当添加 web 引用时,visual studio 将创建代理类,该代理类包含用于访问该 xml web services 的每个公开方法的方法。有关说明,请参见asp”>添加和移除 web 引用。
若要访问 xml web services,您必须创建该代理类的实例,然后与该方法进行交互,就像与类的任何其他实例的方法进行交互一样。
- 包括命名空间 system.threading。您将需要该命名空间来访问 waithandle 类。
- 在要从其中访问 xml web services 的客户端代码中创建代理对象的实例。
- 通过调用 xml web services 方法的 begin 方法,创建到 asyncresult 的接口,它返回实现 iasyncresult 接口的类型。
- 在客户端应用程序中继续进行任何其他的处理。
- 当处于需要来自 xml web services 的结果这种情况时,使用 waithandle 来暂停处理,等待结果。
- xml web services 完成处理并将结果返回给代理后,代理将调用回调函数。
下面显示的代码来自一个 web 应用程序,该应用程序正在访问一个 xml web services,而该应用程序具有一个对该 xml web services 的 web 引用 (translator),该引用包含代理类 (service1),而该代理类具有异步调用 xml web services 的 begin 方法 (beginlongprocess) 和完成调用并获得结果的 end 方法 (endlongprocess)。在此示例中,xml web services 接受字符串,执行某些耗时的转换,然后返回结果。在启动对 xml web services 方法的异步调用之后,客户端应用程序继续一些其他的处理,然后等待 xml web services 返回。当 xml web services 返回后,处理将继续。
visual basic private sub button1_click(byval sender as system.object, _ byval e as system.eventargs) _ handles button1.click dim cservice as new translator.service1() dim ar1 as iasyncresult dim ar2 as iasyncresult call the begin method of the proxy class to initiate the asynchronous call to the xml web service method. ar1 = cservice.beginlongprocess(textbox1.text, nothing, nothing) ar2 = cservice.beginlongprocess(textbox3.text, nothing, nothing) ... keep processing. ... dim wh() as waithandle = {ar1.asyncwaithandle, ar2.asyncwaithandle} waits for both async xml web service calls to finish. waithandle.waitall(wh) retrieves the results by calling the end method of the proxy class. textbox2.text = cservice.endlongprocess(ar1) textbox4.text = cservice.endlongprocess(ar2) end sub // c# private void button1_click(object sender, system.eventargs e) { translator.service1 cservice = new translator.service1(); iasyncresult ar1 = cservice.beginlongprocess(textbox1.text, null, null); iasyncresult ar2 = cservice.beginlongprocess(textbox3.text, null, null); //... // keep processing. //... waithandle[] wh = {ar1.asyncwaithandle, ar2.asyncwaithandle}; // waits for both async xml web service calls to finish. waithandle.waitall(wh); // retrieves the results by calling the end method of the proxy // class. textbox2.text = cservice.endlongprocess(ar1); textbox4.text = cservice.endlongprocess(ar2); }