`
jiq408694711
  • 浏览: 36528 次
  • 性别: Icon_minigender_1
  • 来自: 南京
文章分类
社区版块
存档分类
最新评论

WCF中的异步回调

 
阅读更多
WCF与Web Service不同的是,当我们定义了服务契约的操作时,不管是通过ChannelFactory创建服务代理对象,还是通过SvcUtil的默认方式生成服务代理对象,客户端在调用这些代理对象时,都无法直接实现异步方式的调用。例如,对于如下的服务操作定义:
[OperationContract]
StreamTransferDocument(Documentdocument);

在调用代理对象的方法时,我们无法找到对应于TransferDocument()操作的BeginTransferDocument()和EndTransferDocument()异步方法。

这样的设计使得我们无法通过编程方式异步地调用服务的操作,除非我们在定义服务接口时,直接加入相关操作的异步方法。然而,这又直接导致了服务的设计与方法调用方式之间的耦合。一个好的框架设计要素在于,不管客户端的调用方式(同步或者异步),服务的设计与实现应该是一致的。对于服务的设计者而言,在设计之初,就不应该去考虑服务的调用者调用的方式。换言之,服务操作究竟是否采用异步方式,应该由客户端的调用者决定。因此,所有与异步调用相关的内容应该只与客户端相关。WCF遵循了这一规则。

在我编写的应用程序中,会暴露一个传送文档文件的服务操作。我并不知道也并不关心调用该操作的客户端是否采用异步方式。因此,如上所述的服务操作定义是完全正确的。

那么,客户端究竟应该如何执行异步调用呢?如果采用编程方式获得服务代理对象,这一问题会变得比较糟糕。因为我将服务契约的定义单独形成了一个程序集,并在客户端直接引用了它。然而,在这样的服务契约程序集中,是没有包含异步方法的定义的。因此,我需要修改在客户端的服务定义,增加操作的异步方法。这无疑为服务契约的重用带来障碍。至少,我们需要在客户端维持一份具有异步方法的服务契约。

所幸,在客户端决定采用异步方式调用我所设计的服务操作时,虽然需要修改客户端的服务契约接口,但并不会影响服务端的契约定义。因此,服务端的契约定义可以保持不变,而在客户端则修改接口定义如下:
[ServiceContract]
publicinterfaceIDocumentsExplorerService
{
[OperationContract]
StreamTransferDocument(Documentdocument);

[OperationContract(AsyncPattern
=true)]
IAsyncResultBeginTransferDocument(Documentdocument,
AsyncCallbackcallback,
objectasyncState);

StreamEndTransferDocument(IAsyncResultresult);
}

注意,在BeginTransferDocument()方法上,必须在OperationContractAttribute中将AsyncPattern属性值设置为true,因为它的默认值为false。

调用方式如下:
BasicHttpBindingbinding=newBasicHttpBinding();
binding.SendTimeout
=TimeSpan.FromMinutes(10);
binding.TransferMode
=TransferMode.Streamed;
binding.MaxReceivedMessageSize
=9223372036854775807;

EndpointAddressaddress
=newEndpointAddress
(
"http://localhost:8008/DocumentExplorerService");

ChannelFactory
<IDocumentsExplorerService>factory=
new
ChannelFactory<IDocumentsExplorerService>(binding,address);

m_service
=factory.CreateChannel();

……
IAsyncResultresult
=m_service.BeginTransferDocument(doc,null,null);
result.AsyncWaitHandle.WaitOne();
Streamstream
=m_service.EndTransferDocument(result);

如果采用SvcUtil生成客户端代理文件,可以有更好的方式实现异步,也就是使用SvcUtil的/async开关,例如:
svcutil/asynchttp://localhost:8008/DocumentExplorerService

唯一不足的是,它会不分青红皂白,为所有服务操作都生成对应的异步方法。这样的做法未免过于武断。

合理地利用服务的异步调用,可以有效地提高系统性能,合理分配任务的执行。特别对于UI应用程序而言,可以提高UI的响应速度,改善用户体验。在我编写的应用程序中,下载的文件如果很大,就有必要采用异步方式。

对于异步调用的完成,虽然WCF提供了诸如阻塞、等待轮询等机制,但最好的方式还是使用回调。也就是利用Begin方法参数中的AsyncCallback对象。这是一个委托对象,它的定义如下所示:
publicdelegatevoidAsyncCallback(IAsyncResultar);

利用异步方式执行服务操作,使得服务在执行过程中不会阻塞主线程,当方法执行完成后,通过AsyncCallback回调对应的方法,可以通知客户端服务执行完毕。例如:
//InvokeitAsynchronously
m_service.BeginTransferDocument(m_doc,OnTransferCompleted,null);

//Dosomework;


//callbackmethod
voidOnTransferCompleted(IAsyncResultresult)
{
Streamstream
=m_service.EndTransferDocument(result);
result.AsyncWaitHandle.Close();

lbMessage.Text
=string.Format("Thefile{0}hadbeentransferedsucessfully.",
m_doc.FileName);
}

在调用BeginTransferDocument()方法之后,主线程不会被阻塞,仍然可以继续执行其它工作。而当服务方法执行完毕之后,会自动调用回调方法,执行方法中的内容。

上述实现存在一个问题,就是对于lbMessage控件的操作。由于回调方法并非运行在主线程中,如果回调方法需要更新与异步调用结果相关的界面,例如本例中的lbMessage控件,则需要将回调的调用封送(Marshal)到当前主程序界面的同步上下文中。我们可以使用SynchronizationContext以及它的SendOrPostCallback委托,对调用进行封送:
publicExplorerClientForm()
{
InitializeComponent();
m_synchronizationContext
=SynchronizationContext.Current;
}

privateSynchronizationContextm_synchronizationContext;

则回调方法修改为:
//callbackmethod
voidOnTransferCompleted(IAsyncResultresult)
{
Streamstream
=m_service.EndTransferDocument(result);
result.AsyncWaitHandle.Close();

SendOrPostCallbackcallback
=delegate
{
lbMessage.Text
=string.Format("Thefile{0}hadbeentransferedsucessfully.",
m_doc.FileName);
};
m_synchronizationContext.Send(callback,
null);
}

在调用异步方法时,由于对BeginTransferDocument()和EndTransferDocument()方法的调用可能会在不同的方法体中,因而我将服务代理对象定义为private字段。如果希望将服务对象定义为一个局部变量,可以在调用BeginTransferDocument()方法时,将代理对象传递到方法的asyncState参数中,然后在调用EndTransferDocument()方法之前,通过IAsyncResult获得准确的服务代理对象:
m_service.BeginTransferDocument(m_doc,OnTransferCompleted,m_service);

将m_service作为asyncState对象传入之后,在调用EndTransferDocument()方法之前,就可以根据它先获得服务代理对象:
IDocumentsExplorerServicem_service=result.AsyncStateasIDocumentsExplorerService;
Streamstream
=m_service.EndTransferDocument(result);
//restcodes

(转自博客园:张逸的博客,http://www.cnblogs.com/wayfarer/archive/2007/11/09/954256.html)

分享到:
评论

相关推荐

    wcf服务端异步程序

    - **基于事件的异步模式(EAP)**:这种方式使用回调函数,当异步操作完成时触发事件。 - **基于Task的异步模式(TAP)**:自.NET Framework 4.0起,推荐使用Task类进行异步编程,这种方式更简洁,更符合现代C#...

    WCF异步调用的小例子

    异步调用的调试可能较为复杂,需要关注回调函数和EndXXX方法中的异常处理。异常通常在EndXXX方法中抛出,需要确保正确捕获并处理。 9. **最佳实践** - 使用适当的异步模式,如.NET 4.0及更高版本推荐使用TAP。 -...

    SilverLight和WCF配合异步调用修改xap文件内的配置

    - Silverlight客户端的代码将发起异步调用,获取新配置,并在回调中更新内部状态或资源。 - 可能还包含一些辅助逻辑,比如错误处理、状态管理等。 总的来说,这个主题展示了如何结合Silverlight的富用户体验和WCF的...

    WCF契约回调,客户端通过回调函数,获取服务端的返回值 WCF消息契约

    在本文中,我们将深入探讨Windows Communication Foundation(WCF)中的契约回调及其在实现客户端与服务端双向通信中的应用。WCF是.NET Framework中用于构建分布式应用程序的服务框架,它提供了丰富的功能,包括数据...

    WCF 同步 异步调用 实例

    ### 二、WCF异步调用 异步调用允许客户端在等待服务响应时执行其他任务,提高应用的响应性和性能。WCF提供了两种主要的异步调用模式:基于事件的异步模式(EAP)和基于任务的异步模式(TAP)。这里我们关注基于事件...

    wcf多线程和异步操作

    实现WCF异步服务有多种方式: 1. **回调委托**:这是早期版本.NET中的异步模式,使用IAsyncResult接口,客户端通过传递一个回调委托给开始方法来接收完成通知。 2. **AyncPattern**:基于`BeginInvoke`/`EndInvoke...

    C# WCF 聊天

    WCF支持异步操作,客户端可以使用`BeginXXX`和`EndXXX`方法进行异步调用,并通过回调处理结果。 8. **错误处理和事务** 在聊天应用中,错误处理和事务管理也非常重要。WCF提供了异常处理机制,可以配置服务的行为...

    WCF异步通信

    总结,WCF异步通信提供了单向异步和回调异步两种模式,以适应不同的应用场景。通过使用Service References,开发者可以方便地创建WCF客户端,无缝集成到自己的应用程序中。理解并熟练运用这些机制,能够有效提升WCF...

    Silverlight异步调用WCF接口(非常简单易懂)

    异步调用可能会遇到网络问题或服务端错误,因此在回调方法中要进行适当的错误处理。可以使用`IAsyncResult.AsyncWaitHandle`等待异步操作完成,或者在End方法中捕获可能抛出的异常。 8. **UI更新** 由于...

    C# WCF CallBack Demo

    在WCF中,回调(Callback)是一种高级通信模式,它允许服务主动向客户端发送数据,而不仅仅是响应客户端的请求。这种模式在需要实时数据更新或双向通信的应用场景中非常有用。下面我们将深入探讨C# WCF回调的基本...

    wcfdemo异步测试双工模式

    这些操作可以是异步的,意味着客户端调用服务方法后不会立即阻塞等待结果,而是提供回调方法以处理服务端返回的数据。 异步编程在高并发场景下尤其重要,因为它能避免阻塞线程,提高系统资源利用率。在WCF中,异步...

    asp.net中如何使用WPF回调

    在 ASP.NET 中使用 WCF 回调服务,主要是为了让网页能够实时接收服务端的更新信息,例如在事件管理系统中显示新的事件。 描述中提到的问题在于,ASP.NET 页面的生命周期和 WCF 回调的线程并不一致。当 WCF 回调到达...

    wcf 聊天程序 事件广播

    WCF被用于构建聊天服务,而聊天则要求高效、实时的数据传输和处理,这需要开发者对WCF的异步操作、回调机制以及并发控制有深入的理解。 压缩包内的文件名为“Fish”,这可能是程序的主程序文件或者特定的模块。在...

    WCF 实现订阅发布和请求回复模式

    WCF支持基于回调的异步调用和基于Task的异步编程模型(TAP)。对于基于回调的异步调用,可以使用BeginXXX/EndXXX方法组合;对于TAP,可以使用async/await关键字,这使得代码更易读、易维护。异步调用有助于防止阻塞...

    WCF培训讲稿WCF入门讲稿

    4.3 **双工模式(Duplex)**:允许服务和客户端之间双向通信,类似于回调机制,常用于实时通信。 **5. WCF绑定** 绑定是WCF中的关键概念,它定义了服务与客户端之间的通信方式: 5.1 **基本HTTP绑定**:使用HTTP...

    WCF系列学习源码

    `WCF_双工通信`展示了如何使用回调契约(Callback Contract)来实现这种通信模式,这对于实现如聊天室或实时通知等应用非常有用。 7. **WCF自定义消息**: `WCF_自定义消息`部分展示了如何创建和处理自定义的消息...

    WCF服务实现的WCF双向通讯实例

    这通常包括一个接口(如`IWcfService`),该接口声明了服务将提供哪些操作,并且包含了服务回调的接口(如`IWcfCallback`)。回调接口定义了服务可以调用客户端的方法,以实现双向通信。 2. **配置服务**: 在`Wcf...

    WCF实现双工通讯聊天

    在WCF中,双工通信可以通过使用回调合同来实现。回调合同定义了一个接口,客户端需要实现这个接口,服务端可以通过这个接口调用客户端的方法,从而实现实时通信。 ### 2. 配置方式实现双工通信 创建双工通信服务的...

    VS2010中使用Jquery调用Wcf服务源码

    3. 在成功回调函数中,处理来自WCF服务的响应数据。 4. 不忘处理可能的错误情况,例如网络问题或服务端异常。 在提供的压缩包文件中,"VS2010中使用Jquery调用Wcf服务源码.txt"很可能是包含详细步骤和示例代码的...

Global site tag (gtag.js) - Google Analytics