WCF-异步调用和两种客户端形式
当发布一个服务端之后,客户端可以通过服务端的元数据,用VS2010添加服务引用的方式生成对应的代码。并且可以选择生成相应的异步操作。
WCF实现代码,Add操作延时5秒后再返回结果。
[ServiceContract]
public interface ICalculator
{
[OperationContract]
int Add(int x, int y);
} [ServiceBehavior]
public class Cal : ICalculator
{
public int Add(int x, int y)
{
System.Threading.Thread.Sleep(5000);
return x + y;
}
}
服务寄宿:
static void Main(string[] args)
{
ServiceHost host = new ServiceHost(typeof(kk.Cal));
ServiceMetadataBehavior behavior = new ServiceMetadataBehavior();
behavior.HttpGetUrl = new Uri("http://localhost:6666/meta");
behavior.HttpGetEnabled = true;
host.Description.Behaviors.Add(behavior);
host.AddServiceEndpoint(typeof(kk.ICalculator), new WSHttpBinding(), "http://localhost:6666");
host.Opened+=delegate{Console.WriteLine("Service start!");};
host.Open();
Console.ReadLine();
}
通过ServiceMetadataBehavior的HttpGetUrl可以设置服务的元数据地址,客户端通过这个地址获取到服务的全部信息。
先运行服务端。再按下列步骤在客户端添加服务引用
1.添加服务引用
2.输入服务公布的元数据地址。

3.在高级中选中生成异步操作。再点击2中的前往即可。

一、通过BeginAdd/EndBegin方法调用客户端。
生成成功后通过默认命名空间ServiceReference1获取到客户端,直接调用BeginAdd方法进行异步调用。
static void Main(string[] args)
{
ServiceReference1.CalculatorClient client = new ServiceReference1.CalculatorClient();
client.BeginAdd(1, 2, CallBack, client);
for (int i = 0; i < 100; i++)
{
System.Threading.Thread.Sleep(1000);
Console.WriteLine("{0}", i);
}
} public static void CallBack(IAsyncResult ar)
{
ServiceReference1.CalculatorClient client = ar.AsyncState as ServiceReference1.CalculatorClient;
int result = client.EndAdd(ar);
Console.WriteLine("Result:{0}", result);
}
BeginAdd的前两个参数是契约接口参数。第三个是回调方法,参数类型为IAsyncResult返回值为void的委托。第四个是Object类,这里将client客户端传进去,因为需要在回调方法中调用EndAdd方法得到结果。或者直接把client定义成static成员变量,这样就不需要传入了。
二、通过xxxCompleted添加委托。
static void Main(string[] args)
{
ServiceReference1.CalculatorClient client = new ServiceReference1.CalculatorClient();
client.AddCompleted += delegate(object sender, ServiceReference1.AddCompletedEventArgs e)
{
int[] para = e.UserState as int[];
int result = e.Result;
Console.WriteLine("Result:{0}+{1}={2}", para[0], para[1], result);
};
client.AddAsync(1, 2,new int[]{1,2});
for (int i = 0; i < 100; i++)
{
System.Threading.Thread.Sleep(1000);
Console.WriteLine("{0}", i);
}
}
Completed+=委托是上面的一种变形,直接通过completedEventArgs的Result属性获取结果,UserState获取附加参数。
上面两种异步调用实现效果为:

通过观察在客户端通过添加服务引用生成的代码,
public partial class CalculatorClient : System.ServiceModel.ClientBase<client2.ServiceReference1.ICalculator>, client2.ServiceReference1.ICalculator
发现ServiceReference1.CalculatorClient是继承自
public abstract class ClientBase<TChannel> : ICommunicationObject, IDisposable where TChannel : class
这也意味着我们可以自定义一个Cinent,而不需要去生成而产生很多无用的代码。
using System.ServiceModel;
using System.ServiceModel.Channels;
namespace client2
{
class Program
{
static void Main(string[] args)
{
myClient client = new myClient(new WSHttpBinding(), new EndpointAddress(new Uri("http://localhost:6666")));
int result=client.myChannel.Add(1, 2);
Console.WriteLine("{0}", result);
}
} public class myClient : ClientBase<kk.ICalculator>
{
public myClient(Binding bind, EndpointAddress addr)
: base(bind, addr)
{
}
public kk.ICalculator myChannel
{
get
{
return this.Channel;
}
}
}
}
myClient继承ClientBase并指定泛型契约,重写基类构造函数,指定Binding和EndpointAddress。并通过返回基类Channel属性。构造的时候传入服务端使用的绑定模式,和终结点地址。然后通过myChannel即可获取通道。其实这个ClientBase内部也是通过ChannelFactory<T>实现的,通过信道工厂创建信道,通过信道调用服务端方法。下面代码是通过信道工厂实现。
static void Main(string[] args)
{
ChannelFactory<kk.ICalculator> factory = new ChannelFactory<kk.ICalculator>(new WSHttpBinding(), new EndpointAddress(new Uri("http://localhost:6666")));
kk.ICalculator mychannel=factory.CreateChannel();
mychannel.Add(1, 2);
}
WCF-异步调用和两种客户端形式的更多相关文章
- WCF 异步调用问题
添加引用时生成"勾选允许生成异步操作" Wcf异步调用三种方式: 第一种:直接调用异步方法 var serviceClient = new MyServiceClient(); s ...
- 保姆级别的RabbitMQ教程!一看就懂!(有安装教程,送安装需要的依赖包,送Java、Golang两种客户端教学Case)
保姆级别的RabbitMQ教程!一看就懂!(有安装教程,送安装需要的依赖包,送Java.Golang两种客户端教学Case) 目录 什么是AMQP 和 JMS? 常见的MQ产品 安装RabbitM ...
- DLL调用的两种方式(IDE:VC6.0,C++)
原文:http://www.cnblogs.com/Pickuper/articles/2050409.html DLL调用有两种方式,一种是静态调用,另外一种是动态调用 (一)静态调用 静态调用是一 ...
- Oracle的sql语句的两种判断形式
Oracle的sql语句的两种判断形式 判断当前列同时改动当前列 判断一个情况改动其他值 一类情况详解:实现的是当num这一列的值为3时,就显示好 以此类推 1)case num when 3 the ...
- java SWing事件调用的两种机制
Java(91) /** * java swing中事件调用的两种机制: * (一)响应机制 * (二)回调机制 */ package test; import java.awt.*; impo ...
- JAVA客户端API调用memcached两种方式
1. memcached client for java客户端API:memcached client for java 引入jar包:java-memcached-2.6.2.jar package ...
- WCF异步调用
添加引用服务--高级--选中 生产异步操作 服务端接口操作 [OperationContract]int Add(int a, int b); 客户端: 引用服务:在引用服务时,左下角点击“高级”按钮 ...
- 异步编程的两种模型,闭包回调,和Lua的coroutine,到底哪一种消耗更大
今天和人讨论了一下CPS变形为闭包回调(典型为C#和JS),以及Lua这种具有真正堆栈,可以yield和resume的coroutine,两种以同步的形式写异步处理逻辑的解决方案的优缺点.之后生出疑问 ...
- springcloud 服务调用的两种方式
spring-cloud调用服务有两种方式,一种是Ribbon+RestTemplate, 另外一种是Feign.Ribbon是一个基于HTTP和TCP客户端的负载均衡器,其实feign也使用了rib ...
随机推荐
- c# 利用t4模板,自动生成Model类
我们在用ORM(比如dapper)的时候,很多时候都需要自己写Model层(当然也有很多orm框架自带了这种功能,比如ef),特别是表里字段比较多的时候,一个Model要写半天,而且Model如果用于 ...
- C#基础之流程控制语句详解
C#程序的执行都是一行接一行.自上而下地进行,不遗漏任何代码.为了让程序能按照开发者所设计的流程进行执行,必然需要进行条件判断.循环和跳转等过程,这就需要实现流程控制.C#中的流程控制包含了条件语句. ...
- django系列8.5--使用装饰器(视图函数中)实现用户登录状态检验
views.py def session_auth(fn): def inner(request,*args,**kwargs): status = request.session.get('sess ...
- Java面试题(全)--视频系列
此系列为面试笔试题的视频讲解,以下均为超链接,点击即可进入每个知识点的讲解. Java面试题01.面试的整体流程 Java面试题02.java的垮平台原理 Java面试题03.搭建一个java的开发环 ...
- Thread.sleep(1000)
public class Wait { public static void main(String[] args) { System.out.println(System.currentTimeMi ...
- djngo 1.9版本以后 Foreignkey() 字段 第二个参数 on_delete 必不可少, mysql 外键可以为空
一.外键的删除 1.常见的使用方式(设置为null) class BookModel(models.Model): """ 书籍表 """ ...
- spring-mybatis源码追踪
启用一个扫描类 <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"> <propert ...
- Linux(Ubuntu)配置问题集
Cannot set LC_CTYPE to default locale: No such file or directory 安装Ubuntu Server版本(不带桌面环境)时,如果安装时的语言 ...
- VMware Workstation 14 pro License Keys
AC5XK-0ZD4H-088HP-9NQZV-ZG2R4 CG54H-D8D0H-H8DHY-C6X7X-N2KG6 ZC3WK-AFXEK-488JP-A7MQX-XL8YF ZC5XK-A6E0 ...
- 使用Docker构建jdk1.8镜像
一.下载centos镜像 下载自己需要的版本TAG,详见: docker安装指定版本TAG的镜像 $ sudo docker pull centos:centos7 二.下载jdk1.8,并上传到/u ...