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 ...
随机推荐
- Mysql修改字段类型
mysql 修改字段长度 alter table news modify column title varchar(130); alter table 表名 modify column 字段名 类型 ...
- 函数IsValid()
功能:检查对象变量是否已经实例化,即实例变量的值是否是个有效的对象句柄. 语法:IsValid(objectname) 参数:objectname:要检查的对象名. 返回值:Boolean.如果指定对 ...
- 记一次MBR锁机病毒分析
有一天,在机缘巧合之下我获得了一个锁机软件(是多巧合阿喂!),然后兴高采烈的把它拖入了虚拟机里蹂躏(>_<!). 很巧,软件有虚拟机检测... Emmmm好吧,随便过一下... 我用的虚拟 ...
- Java中类变量和实例变量的初始化
1. 类变量和实例变量 类变量即类成员变量中的静态变量,它们可以通过类名来直接访问. 实例变量是类成员变量中的非静态变量,只有在实例化对象之后通过对象来访问. 2. 空间分配的时间不同 类变量是在类加 ...
- 【Qt开发】实现系统托盘,托盘菜单,托盘消息
概述 系统托盘就是在系统桌面底部特定的区域显示运行的程序.windows在任务栏状态区域,linux在布告栏区域.应用程序系统托盘功能,是比较普遍的功能,本篇将详细的介绍如何实现该功能. 演示Demo ...
- 插入排序 思想 JAVA实现
已知一个数组 60.28.41.39.6 .18 .14.28.49.31 利用插入排序算法进行排序 插入排序是一个运行时间为O(N²)的排序算法. 算法思想 60.28.41.39.6 .18 . ...
- PHP下的浮点运算不准的解决办法
首先看一段代码: 首先看一段代码: <?php $a = 0.1; $b = 0.7; var_dump(($a + $b) == 0.8); 打印出来的值居然为 boolean false P ...
- ownCloud问题处理server replied 423 Locked to
打开owncloud 数据库备份:oc_file_locks表(备份免错哦)然后清空该表,客户端同步一次,故障解决 owncloud大的数据无法同步..
- 算法图解学习笔记01:二分查找&大O表示法
二分查找 二分查找又称折半查找,其输入的必须是有序的元素列表.二分查找的基本思想是将n个元素分成大致相等的两部分,取a[n/2]与x做比较,如果x=a[n/2],则找到x,算法中止:如果x<a[ ...
- Haskell优雅的快排实现
说得快速排序,基本是常用的排序当中速度最快的排序了,之前也用C和Java实现过,但是过程十分痛苦,更重要的是写完代码只记得过程却对实质的过程觉得隔了一层纱,有种说不出的感觉.刚刚看一下Haskell实 ...