您的位置:首页 > 编程语言

【转载】WCF热门问题编程示例(4):WCF客户端如何异步调用WCF服务?

2011-11-07 10:11 706 查看

WCF - IsOneWay 和异步的区别

http://www.rainsts.net/article.asp?id=460

在某些博客文章里,直接将 IsOneWay 称为 "异步方法"。虽然多数时候不会对开发带来什么问题,但深究起来,这两者是不同的。接下来,我们做个试验。将同一个服务契约分别用 IsOneWay 和异步进行实现,客户端使用多线程模拟并发调用,并使用 ServiceThrottlingBehavior (也可以使用 InstanceContextMode.Single) 进行并发控制。注意对比输出结果,我们就会发现其不同之处。

IsOneWay版本

[ServiceContract]
public interface IService
{
[OperationContract(IsOneWay=true)]
void Test();
}

public class MyService : IService
{
public void Test()
{
Console.WriteLine("Service BeginAdd:{0}", DateTime.Now);
Thread.Sleep(5000);
Console.WriteLine("Service EndAdd:{0}", DateTime.Now);
}
}

public class WcfTest
{
public static void Test()
{
AppDomain.CreateDomain("Server").DoCallBack(delegate
{
ServiceHost host = new ServiceHost(typeof(MyService));
host.AddServiceEndpoint(typeof(IService), new WSHttpBinding(), "http://localhost:8080/myservice");

ServiceThrottlingBehavior throttling = new ServiceThrottlingBehavior();
throttling.MaxConcurrentInstances = 1;
host.Description.Behaviors.Add(throttling);

host.Open();
});

for (int i = 0; i < 3; i++)
{
new Thread(delegate()
{
IService channel = ChannelFactory<IService>.CreateChannel(new WSHttpBinding(),
new EndpointAddress("http://localhost:8080/myservice"));

using (channel as IDisposable)
{
Console.WriteLine("Client {0} BeginAdd:{1}",
Thread.CurrentThread.ManagedThreadId, DateTime.Now);

channel.Test();

Console.WriteLine("Client {0} BeginAdd End:{1}",
Thread.CurrentThread.ManagedThreadId, DateTime.Now);
}
}).Start();
}
}
}


输出:
Client 15 BeginAdd:2007-4-19 10:51:39
Client 14 BeginAdd:2007-4-19 10:51:39
Client 16 BeginAdd:2007-4-19 10:51:39
Client 16 BeginAdd End:2007-4-19 10:51:40
Service BeginAdd:2007-4-19 10:51:40
Service EndAdd:2007-4-19 10:51:45
Client 14 BeginAdd End:2007-4-19 10:51:46
Service BeginAdd:2007-4-19 10:51:46
Service EndAdd:2007-4-19 10:51:51
Service BeginAdd:2007-4-19 10:51:51
Client 15 BeginAdd End:2007-4-19 10:51:51
Service EndAdd:2007-4-19 10:51:56

异步版本

[ServiceContract]
public interface IService
{
[OperationContract]
void Test();

[OperationContract(AsyncPattern = true)]
IAsyncResult BeginTest(AsyncCallback callBack, object state);

void EndTest(IAsyncResult ar);
}

public class MyService : IService
{
public void Test()
{
Console.WriteLine("Service BeginAdd:{0}", DateTime.Now);
Thread.Sleep(5000);
Console.WriteLine("Service EndAdd:{0}", DateTime.Now);
}

public IAsyncResult BeginTest(AsyncCallback callBack, object state)
{
throw new Exception("The method or operation is not implemented.");
}

public void EndTest(IAsyncResult ar)
{
throw new Exception("The method or operation is not implemented.");
}
}

public class WcfTest
{
public static void Test()
{
AppDomain.CreateDomain("Server").DoCallBack(delegate
{
ServiceHost host = new ServiceHost(typeof(MyService));
host.AddServiceEndpoint(typeof(IService), new WSHttpBinding(), "http://localhost:8080/myservice");

ServiceThrottlingBehavior throttling = new ServiceThrottlingBehavior();
throttling.MaxConcurrentInstances = 1;
host.Description.Behaviors.Add(throttling);

host.Open();
});

for (int i = 0; i < 3; i++)
{
new Thread(delegate()
{
IService channel = ChannelFactory<IService>.CreateChannel(new WSHttpBinding(),
new EndpointAddress("http://localhost:8080/myservice"));

using (channel as IDisposable)
{
Console.WriteLine("Client {0} BeginAdd:{1}",
Thread.CurrentThread.ManagedThreadId, DateTime.Now);

IAsyncResult ar = channel.BeginTest(null, null);

Console.WriteLine("Client {0} BeginAdd End:{1}",
Thread.CurrentThread.ManagedThreadId, DateTime.Now);

channel.EndTest(ar);
}
}).Start();
}
}
}


输出:
Client 14 BeginAdd:2007-4-19 10:53:56
Client 16 BeginAdd:2007-4-19 10:53:56
Client 15 BeginAdd:2007-4-19 10:53:56
Client 16 BeginAdd End:2007-4-19 10:53:56
Client 14 BeginAdd End:2007-4-19 10:53:56
Client 15 BeginAdd End:2007-4-19 10:53:56
Service BeginAdd:2007-4-19 10:53:59
Service EndAdd:2007-4-19 10:54:04
Service BeginAdd:2007-4-19 10:54:04
Service EndAdd:2007-4-19 10:54:09
Service BeginAdd:2007-4-19 10:54:09
Service EndAdd:2007-4-19 10:54:14

通过对比,我们发现异步方法 BeginXXX 调用并不受并发控制影响,调用后直接返回控制权;而 IsOneWay 则不同,它被阻塞直到服务方法获得执行才会返回(当然,没有等待服务方法执行完成)。这点区别在处理并发性能上,可能带来不同的效果,了解 IsOneWay 和异步的差别能让我们避免一些意外的问题。

------------------------------------------------------------------------

WCF热门问题编程示例(4):WCF客户端如何异步调用WCF服务?

How to call WCF Service asynchronously?

【1】问题描述:


WCF客户端如何异步调用WCF服务?

How to call WCF Service asynchronously?


关于WCF如何实现异步调用的问题,论坛上出现了很多帖子,也有很多讨论的文章,包括MSDN也给出了详细的学习资料和示例代码。但是很多资料过于笼统,MSDN的例子有点复杂。而我们实际项目中,要实现的需求,往往十分简单,就是要在客户端实现对于WCF服务操作的异步调用,也就是call WCF Service asynchronously。

【2】资料收集:

这里收集一些关于WCF异步调用比较有用的参考资料和问题连接。其实关于WCF客户端异步调用WCF服务的帖子很多,很多讨论反而走了弯路,我们想要的就是简单的异步调用WCF服务操作的例子。这里给出一些比较有价值的参考资料:

如何:以异步方式调用 WCF 服务操作
How to: Call WCF Service Operations Asynchronously
ASP.NET异步调用WCF

【3】异步调用:

这里需要了解的一个重要概念就是异步调用机制,在.NET里的异步调用机制是十分重要的一个知识点。另外就是回调的概念,这里也会使用到。.Net中的异步调用实际使用的是异步委托。异步委托将要执行的方法提交到.Net的线程池,由线程池中的线程来执行异步调用。

【4】解决办法:

知道了.NET异步调用的机制以后,我们就可以来尝试解决这个问题。这里本质是一样的。但编码的过程有点差别。

当然,最容易理解的就是我们自己写代码来实现异步调用。
其次,Visual Studio 2008已经提供了这个支持。这个更加方便。我给出截图:





我们这里给出的示例过程呢,是基于手写客户端异步调用代码的。仅供参考,当然你也可以使用Visual Studio来帮助完成这个过程。

【5】服务端代码:

服务端代码的实现十分简单,这里我们依然使用早期WCF的代码,定义一个IWCFService契约,包含一个操作SayHello,这个操作实现的功能,就是接受一name参数,然后打印返回字符串给客户端,为了演示,这里会打印出时间信息。具体代码如下:


//1.服务契约
[ServiceContract(Namespace = "“%3E%3Cfont%20color=]http://www.cnblogs.com/frank_xl/")]
public interface IWCFService
{
//操作契约
[OperationContract]
string SayHello(string name);

}
//2.服务类,继承接口。实现契约
public class WCFService : IWCFService
{
//实现接口定义的方法
public string SayHello(string name)
{
Console.WriteLine("Call Operation at {0}", DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:ffff"));
return string.Format("Hello {0}",name);
}
}


【6】客户端代码:

客户端的实现大部分代码与之前的没有区别,为了方便,我们可以在Visual Studio生成的代码基础上修改。当然也完全可以自己重写。 这里比较重要的一点,就是要实现异步调用需要的契约以及异步调用的方法。

【6.1】首先这里要实现支持异步调用的契约:


public interface IWCFService {
[System.ServiceModel.OperationContractAttribute(Action="http://www.cnblogs.com/frank_xl/IWCFService/SayHello", ReplyAction="“%3E%3Cfont%20color=]http://www.cnblogs.com/frank_xl/IWCFService/SayHelloResponse")]
string SayHello(string name);

[OperationContract(AsyncPattern = true)]
IAsyncResult BeginSayHello(string name, AsyncCallback callback, object asyncState);

string EndSayHello(IAsyncResult result);
}


【6.2】支持异步调用的客户端代理类:

我们要在自己的WCFClient代理类里继承这个契约,并提供异步操作调用的实现,具体代码如下:


public string SayHello(string name) {
return base.Channel.SayHello(name);
}

public IAsyncResult BeginSayHello(string name, AsyncCallback callback, object asyncState)
{
return Channel.BeginSayHello(name, callback, asyncState);
}

public string EndSayHello(IAsyncResult result)
{
return Channel.EndSayHello(result);
}


【6.3】测试客户端代码:

客户端代码还有一点比较重要的地方,就是实现异步回调方法,这里我们在调用BeginSayHello方法的时候,不需要等待结果的返回,我们自己希望,WCF在调用完SayHello操作以后,执行一些必要的工作,比如显示返回结果,或者对返回的数据做更深入的处理,这里都需要使用到回调函数。我们也可以在回调方法里,来关闭WCF客户端,以释放资源。

例子代码如下:


/// <summary>
/// This class is defined for WCF Async Call Test
/// </summary>
public class WCFClientTest
{
static public WCFServiceClient wcfServiceProxy = null;
static void Main(string[] args)
{
Console.ForegroundColor = ConsoleColor.Yellow;

WCFClientTest client = new WCFClientTest();
wcfServiceProxy = new WCFServiceClient("WSHttpBinding_IWCFService");
string name = "Frank Xu Lei";
Console.WriteLine("Client Async Call begin at {0}",DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:ffff"));
//开始异步调用Begin Call
wcfServiceProxy.BeginSayHello(name, client.OnCompletion, null);

Console.WriteLine("press ENTER to exit…");
Console.Read();

}
////回调方法,Callback method
void OnCompletion(IAsyncResult result)
{
string value = wcfServiceProxy.EndSayHello(result);
Console.WriteLine("Returned value is {0} at {1}",value, DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:ffff"));
result.AsyncWaitHandle.Close();
wcfServiceProxy.Close();
Console.WriteLine("Asynchronous Calls is finished");
}
}


【7】运行结果:

这里启动宿主Host程序,然后运行WCF客户端即可,我们可以看到宿主端和客户端,异步调用操作的时间差别,打印窗口的截图如下:



内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐