WCF与Web Service不一样的是,当咱们定义了服务契约的操做时,不论是经过ChannelFactory建立服务代理对象,仍是经过SvcUtil的默认方式生 成服务代理对象,客户端在调用这些代理对象时,都没法直接实现异步方式的调用。例如,对于以下的服务操做定义:编程
[OperationContract]
Stream TransferDocument(Document document);
在调用代理对象的方法时,咱们没法找到对应于TransferDocument()操做的BeginTransferDocument()和EndTransferDocument()异步方法。 框架
这 样的设计使得咱们没法经过编程方式异步地调用服务的操做,除非咱们在定义服务接口时,直接加入相关操做的异步方法。然而,这又直接致使了服务的设计与方法 调用方式之间的耦合。一个好的框架设计要素在于,无论客户端的调用方式(同步或者异步),服务的设计与实现应该是一致的。对于服务的设计者而言,在设计之 初,就不该该去考虑服务的调用者调用的方式。换言之,服务操做到底是否采用异步方式,应该由客户端的调用者决定。所以,全部与异步调用相关的内容应该只与 客户端相关。WCF遵循了这一规则。 异步
在我编写的应用程序中,会暴露一个传送文档文件的服务操做。我并不知道也并不关心调用该操做的客户端是否采用异步方式。所以,如上所述的服务操做定义是彻底正确的。 async
那 么,客户端究竟应该如何执行异步调用呢?若是采用编程方式得到服务代理对象,这一问题会变得比较糟糕。由于我将服务契约的定义单独造成了一个程序集,并在 客户端直接引用了它。然而,在这样的服务契约程序集中,是没有包含异步方法的定义的。所以,我须要修改在客户端的服务定义,增长操做的异步方法。这无疑为 服务契约的重用带来障碍。至少,咱们须要在客户端维持一份具备异步方法的服务契约。 ide
所幸,在客户端决定采用异步方式调用我所设计的服务操做时,虽然须要修改客户端的服务契约接口,但并不会影响服务端的契约定义。所以,服务端的契约定义能够保持不变,而在客户端则修改接口定义以下: 性能
[ServiceContract]
public
interface
IDocumentsExplorerService
{
[OperationContract]
Stream TransferDocument(Document document);
[OperationContract(AsyncPattern
=
true
)]
IAsyncResult BeginTransferDocument(Document document,
AsyncCallback callback,
object
asyncState);
Stream EndTransferDocument(IAsyncResult result);
}
注意,在BeginTransferDocument()方法上,必须在OperationContractAttribute中将AsyncPattern属性值设置为true,由于它的默认值为false。
调用方式以下:
等待和轮询等机制,但最好的方式仍是使用回调。也就是利用Begin方法参数中的AsyncCallback对象。这是一个委托对象,它的定义以下所示:
spa
利用异步方式执行服务操做,使得服务在执行过程当中不会阻塞主线程,当方法执行完成后,经过AsyncCallback回调对应的方法,能够通知客户端服务执行完毕。例如: 线程
//
Invoke it Asynchronously
m_service.BeginTransferDocument(m_doc,OnTransferCompleted,
null
);
//
Do some work;
//
callback method
void
OnTransferCompleted(IAsyncResult result)
{
Stream stream
=
m_service.EndTransferDocument(result);
result.AsyncWaitHandle.Close();
lbMessage.Text
=
string
.Format(
"
The file {0} had been transfered sucessfully.
"
,
m_doc.FileName);
}
在调用BeginTransferDocument()方法以后,主线程不会被阻塞,仍然能够继续执行其它工做。而当服务方法执行完毕以后,会自动调用回调方法,执行方法中的内容。
上 述实现存在一个问题,就是对于lbMessage控件的操做。因为回调方法并不是运行在主线程中,若是回调方法须要更新与异步调用结果相关的界面,例如本例 中的lbMessage控件,则须要将回调的调用封送(Marshal)到当前主程序界面的同步上下文中。咱们可使用SynchronizationContext以及它的SendOrPostCallback委托,对调用进行封送:
则回调方法修改成:
TransferDocument()和EndTransferDocument()方法的调用可能会在不一样的方法体中,于是我将服务代理对象定义为private字段。若是但愿将服务对象定义为一个局部变量,能够在调用BeginTransferDocument()方法时,将代理对象传递到方法的asyncState参数中,而后在调用EndTransferDocument()方法以前,经过IAsyncResult得到准确的服务代理对象:
将m_service做为asyncState对象传入以后,在调用EndTransferDocument()方法以前,就能够根据它先得到服务代理对象:
设计
public
ExplorerClientForm()
{
InitializeComponent();
m_synchronizationContext
=
SynchronizationContext.Current;
}
private
SynchronizationContext m_synchronizationContext;
//
callback method
void
OnTransferCompleted(IAsyncResult result)
{
Stream stream
=
m_service.EndTransferDocument(result);
result.AsyncWaitHandle.Close();
SendOrPostCallback callback
=
delegate
{
lbMessage.Text
=
string
.Format(
"
The file {0} had been transfered sucessfully.
"
,
m_doc.FileName);
};
m_synchronizationContext.Send(callback,
null
);
}
在调用异步方法时,因为对Begin代理
m_service.BeginTransferDocument(m_doc,OnTransferCompleted,m_service);
IDocumentsExplorerService m_service
=
result.AsyncState
as
IDocumentsExplorerService;
Stream stream
=
m_service.EndTransferDocument(result);
//
rest codes
BasicHttpBinding binding
=
new
BasicHttpBinding();
binding.SendTimeout
=
TimeSpan.FromMinutes(
10
);
binding.TransferMode
=
TransferMode.Streamed;
binding.MaxReceivedMessageSize
=
9223372036854775807
;
EndpointAddress address
=
new
EndpointAddress
(
"
http://localhost:8008/DocumentExplorerService
"
);
ChannelFactory
<
IDocumentsExplorerService
>
factory
=
new
ChannelFactory
<
IDocumentsExplorerService
>
(binding,address);
m_service
=
factory.CreateChannel();
……
IAsyncResult result
=
m_service.BeginTransferDocument(doc,
null
,
null
);
result.AsyncWaitHandle.WaitOne();
Stream stream
=
m_service.EndTransferDocument(result);
若是采用SvcUtil生成客户端代理文件,能够有更好的方式实现异步,也就是使用SvcUtil的/async开关,例如:
svcutil
/
async http:
//
localhost:8008/DocumentExplorerService
惟一不足的是,它会不分青红皂白,为全部服务操做都生成对应的异步方法。这样的作法未免过于武断。
合理地利用服务的异步调用,能够有效地提升系统性能,合理分配任务的执行。特别对于UI应用程序而言,能够提升UI的响应速度,改善用户体验。在我编写的应用程序中,下载的文件若是很大,就有必要采用异步方式。
对于异步调用的完成,虽然WCF提供了诸如阻塞、
public
delegate
void
AsyncCallback(IAsyncResult ar);