毕业设计-基于WCF的即时通讯软件的设计与实现.pdf

上传人:曲**** 文档编号:96293695 上传时间:2023-10-16 格式:PDF 页数:67 大小:2.04MB
返回 下载 相关 举报
毕业设计-基于WCF的即时通讯软件的设计与实现.pdf_第1页
第1页 / 共67页
毕业设计-基于WCF的即时通讯软件的设计与实现.pdf_第2页
第2页 / 共67页
点击查看更多>>
资源描述

《毕业设计-基于WCF的即时通讯软件的设计与实现.pdf》由会员分享,可在线阅读,更多相关《毕业设计-基于WCF的即时通讯软件的设计与实现.pdf(67页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。

1、基于WCF的即时通讯软件的设计与实现摘要:介绍了 Microsoft用于构建分布式面向服务架构系统的新一代框架WCF的体系结构 与技术要素,并通过开发一套即时通信软件展现了基于WCF构架开发分布式应用程序的编 程步骤与技巧。关键词:WCF;SOA;分布式;即时通信、,.一?一 一、刖5自从在微软提出.NET战略以来,就针对建立企业级的分布式应用先后推出了一系列产 品和技术,包括:ASP.NET Web服务、.NET RemotingMessage Queuing 以及Enterprise Service 等。这些技术为基于微软技术的软件研发人员开发分布式应用提供了很大的便利,同时也各 自存在着

2、一些不足。WCF(Windows Communication Foundation)作为微软基于SOA所推出 的.NET平台下的新一代框架产品集成了现有技术的优点,代表了未来软件架构设计与开发 的发展方向。因此,掌握并能在未来应用中合理运用WCF技术,对于程序员特别是基于微 软技术开发的程序员而言是十分必要的。基于此,文章通过介绍一套即时通信软件的具体开 发过程来展现基于WCF技术的分布式软件研发的基本步骤与高级技巧。二、WCF概述对于一个好的分布式系统来讲,设计时应当考虑到异构性、开放性、安全性、可扩展性、故障处理、并发性以及透明性等问题。基于SOAP的Web Service可以实现异构环境

3、的互操作性,保证了跨平台的通信。利用WSE(Web Service Enhancements)可以为ASMX提供安 全性的保证。.NETRemoting具有丰富的扩展功能,可以创建定制的信道、格式化器和代理 程序。Enterprise Service(COM+)提供了对事务的支持,其中还包括分布式事务,可实现 故障的恢复。MSMQ可以支持异步调用、脱机连接、断点连接等功能,利用消息队列支持 应用程序之间的消息传递。从功能角度来看,WCF整合了 ASMX.Net Remoting Enterprise Service.WSE以及MSMQ等现有技术的优点,它提供了一种构建安全可靠的分布式面向服 务

4、系统的统一的框架模型,使软件研发人员在开发分布式应用时变得更加轻松。1.面向服务既然WCF是一套面向服务的框架,服务自然便是WCF中最为重要的概念。服务是指 暴露在外的一系列功能的集合,面向服务则是指一套构建“面向服务程序”的抽象原则以及 最优方法。对于业务逻辑的理解,传统的编程方式认为应将业务逻辑封装为对象,该对象提 供了与业务相关的一些功能;而基于WCF的程序设计却更多的是考虑如何提供服务以及消 费服务。与面向组件服务程序类似,基于SOA的应用程序将服务封装到了单个逻辑程序当 中,如图1所示。图1封装服务的SOA应用程序逻辑图2.WCF体系结构WCF拥有一个非常灵活的分层体系结构,分布式应

5、用程序可以使用高级API或者低级API编写。高级API或者服务层可以用于调用方法和事件。服务层把这些高级的抽象代码转换为消息,以使用低级API上的信道和端口。图2中显示了 WCF应用程序的各个层。图2 WCF体系结构图WCF提供了对可靠性、事务性、并发管理、安全性以及实例激活等技术的有力支持,而这些支持均依赖于如图3所示的WCF构架。在客户端,分布式应用通过一个代理来转发对宿主端所提供服务的调用,而代理拥有和 服务相同的操作接口,另外还有一些附加的代理管理方法。这也就意味着客户端从来不会直 接调用服务,即便这个服务就在本机的内存中。当客户端代理接收到来自客户端的调用请求 后,它将消息通过信道链

6、向下传递。每个信道都会执行相应的消息的调用前处理,例如对消 息的编码、提供可靠的会话、对消息进行加密等。客户端的最后一个信道则是传输信道,根 据配置的传输方式发送消息给宿主。在宿主端,消息同样通过信道链进行传输。与客户端信道相对应,宿主端信道也会对消 息执行相应的宿主端的调用前处理,例如对消息的解码、提供会话管理、对消息进行解密等。宿主端的最后一个信道则负责将消息发送给消息分发器(Dispatcher),由分发器负责调用 服务的实例。图3 WCF构架示意图3.WCF的基本技术要素作为基于SOA的一个框架产品,WCF实际上是构建了一个在互联系统中实现各个应用 程序之间通信的分布式框架。它使得系统

7、构架师与开发人员在构建分布式系统时,能将更多 的精力投入到与系统的业务逻辑本身的设计上来,而无需过多的考虑底层通信的实现及相关 问题。WCF最核心的部分是能够快捷的创建一个服务,一个WCF服务端框架由宿主、端点以及服务类三部分所组成,如图4所示。图4WCF服务框架宿主(Host),即承载WCF Service运行的环境。可用的宿主环境包括:(1)自承载方式:在控制台应用程序与基于WinForm的应用程序中都可以使用这种方式;(2)系统服务方式:服务可以随着操作系统的启动而自动启动;(3)IIS方式:与WebServices的部署方式类似,由请求消息来激活服务,但只支持HTTP 方式的绑定;(4

8、)WAS(Windows Process Activation Service)方式:这个宿主是 IIS7 的一部分,只有 Windows Vista和Windows Server 2008提供默认支持,它支持几乎所有的通讯协议并提供 了应用程序池、循环回收、空闲管理、身份管理、隔离等强大的功能。服务类(Service Class)是指一个标记了一些WCF特有的属性的类,它包含了对服务 的业务逻辑的具体实现。端点(Endpoints)是WCF实现通信的核心要素,客户端和服务端都通过端点来交换消息,WCF允许我们为服务添加多个绑定和端点。端点由地址(Address)、绑定(Binding)以 及

9、契约(Contract)三部分组成,如图5所示。在WCF中,类ServiceEndpoint代表了一个 Endpoint,在类中包含的 Endpoint Address,Binding,ContractDescri ption 类型分别对应端点 中的地址、绑定以及契约。图5端点构成图地址:每个服务都会关联到一个唯一的地址,因此地址定位和唯一标志了一个端点,其 主要提供了两个重要信息:服务位置以及传送协议。在WCF中,地址由 System.ServiceModel.EndpointAddress 对象来表示,其包括 URI Identity Headers 三个要素。绑定:绑定提供了一种可设置的

10、方式来选择传输协议、消息编码、通讯模式、可靠性、安全 性、事务传播以及交互方式等。例如在传输协议上可以选择HTTP/HTTPS、TCP、P2P、IPC 甚至是MSMQ等方式。消息编码上可以选择使用纯文本方式来确保互操作能力,或者选择 二进制编码来优化性能,或者使用MTOM来提高负载能力,甚至是自定义编码方式。WCF 中提供了 BasicHttpBinding NetTcpBinding、NetPeerTcpBinding NetNamedPipeBinding WSHttpBinding、WSFederationHttpBinding WSDualHttpBinding、NetMsmqBind

11、ing 以及 MsmqlntegrationBinding九种标准类型的绑定。契约:契约是用来描述服务功能的一种平台中立的标准方式,WCF所有服务都需要实 现一个或多个契约。WCF定义了四种类型的契约:(1)服务契约(Service Contracts):定义了客户端可以使用哪些服务操作。(2)数据契约(Data Contracts):定义了服务传输的数据类型。WCF定义了一些隐式数 据契约,比如int string等,但更多时候需要使用DataContractAttribute显式定义那些自 定义类型数据的数据契约。(3)错误处理契约(Fault Contracts):定义了服务引发的错误信

12、息,以及如何将这些异常 传递给客户端。(4)消息契约(Message Contracts):允许直接操控服务的消息内容和格式。一般情况下,应当用接口来定义服务契约,尽管我们也可以使用类。将服务契约定义为 接口基于如下几点优点:(1)便于契约的继承,不同根的类型可以自由实现相同的契约;(2)同一服务类型可以同时实现多个契约;(3)类似于接口隔离原则,可以随时修改服务类型的实现而不影响其它实现;(4)便于制定版本升级策略,新、旧版本的服务契约可以同时使用而互不影响。在WCF中,对于自承载的服务,端点的相关的信息可以有代码实现与配置文件两种定 义方式。而对于IIS承载服务,端点的相关的信息一般定义在

13、虚拟根目录下的Web.Config 文件中。一般来讲,使用配置文件来定义端点相关信息是更为灵活一、更为推荐的一种方式,其可以在不修改代码、不重新发布系统的情况下对服务的地址、绑定和契约等参数进行修改(因为修改config类型文件的内容是不需要重新编译和重新部署的)。在下面的代码中具体说明了如何定义宿主端的端点相关信息。其中地址为 http:/localhost:8080/HelloService,契约为 WCFServiceHello 命名空间下的 IHello 接口,绑定 采用的是WSHttpBinding方式。值得注意的是,代码中的HelioService为相对地址,http:/local

14、host:8080/提供的是基址,当然去掉基址直接将address设为 http:/localhost:8080/HelloService 也是可以的。代码中还添加了名为 MyServiceTypeBehaviors 的行为配置,其将serviceMetadata节中的httpGetEnabled属性设为了 true,目的是为了自动透 过HTTP-GET发布服务的元数据。WCF提供的另外一种发布元数据的方式是使用专门的 MEX端点。在接下来的宿主代码中,只需要简单的创建ServiceHost类型的对象,并利用其实例方法Open启动服务应用程序即可,简要代码如下所示:using(ServiceH

15、ost host=new ServiceHost(typeof(WCFServiceHello.HelloWorld)Console.WriteLine(HelloService has been started.);host.Open();Console.ReadKeyO;)客户端和服务之间通过消息交换来完成方法调用和数据传递,而在WCF中定义了 3 种消息交换模式,如图6所示。(1)OneWay:这种消息交换模式在调用方法后会立即返回而不需要等待服务端的消息返 回。(2)Request/Reply:这种消息交换模式属于同步调用。在调用服务方法后需要等待服务端 的消息返回。(3)Duplex

16、:这种消息交换模式具有客户端与服务端双向通信的功能,同时它的实现还可 以使消息交换具有异步回调的作用。图6 WCF中的3种消息交换模式在设置完宿主端端点之后,同样也必须为分布式应用程序定义客户端的端点,而且只有 当客户端的端点与宿主端的某个端点相互匹配时,客户端的请求才能被宿主端所监听到。如 果服务提供了发布元数据,那么利用.NET Framework 3.0 SDK所提供的SvcUtil.exe工具可 以很轻松的自动生成与宿主端对应的客户端代理以及客户端配置文件。比如,运行宿主端应 用程序,然后打开Visual Studio 2005命令提示符,键入SvcUtil http:/localho

17、st:8080,便可 以在当前目录下得到客户端代理文件HelloWorld.cs与客户端配置文件output.config。另外一 种简便直观的可视化工具是SDK所附带的SvcConfigEditor.exe(C:Program FilesMicrosoftSDKsWindowsv6.0Abin目录下,XP系统),使用这个工具可以非常方便地创建或修改宿 主端和客户端的配置文件。生成好客户端代理与配置文件后,在代码中直接使用客户端代理对象即可。using(HelloClient client=new HelloClient()(client.HelloO;)Console.ReadKeyO;另外

18、一种创建客户端代理的方式是使用ChannelFactory动态的来创建。虽然WCF提供 了这种方式,但是在实际开发中并不推荐使用它,毕竟ChannelFactory直接依赖于契约,而这恰恰违背了 SOA中边界隔离的原则。利用服务器端与客户端之间的Channel来创建客 户端代理的代码举例如下:ServiceEndpoint httpendpoint=new ServiceEndpoint(ContractDescription.GetContract(typeof(IHello),new WSHttpBindingO,new EndpointAddress(http:/localhost:80

19、80/HelloService);using(ChannelFactory factory=new ChannelFactory(httpendpoint)(IHello service=factory.CreateChannel();service.Hello();)Console.ReadKeyO;三、软件的分析与设计软件主要的功能是初步实现基于WCF的局域网内的实时通信,以面向服务为指导思想 将具体开发过程分为即时通信服务的设计与实现、宿主的设计与实现以及即时通信客户端的 设计与实现三部分,使得应用程序具有较好的安全性、并发性、可扩展性以及可维护性。1.服务的设计服务的设计包括了通用模块

20、Common.dll的设计、服务契约IChat的定义与实现、客户 端的回调接口 IChatCallback的定义三个部分。通用模块Common.dll主要包括对聊天者类型Person的定义,其包括name(聊天者的名称 或代号)以及ImageURL(聊天者选择的头像图片的存储路径)两个私有字段及相应的属性 访问器。由于Person类型是自定义数据类型,因此必须加上DataContractAttribute来显式 定义它。Person类的实现核心代码如下:DataContractpublic class Person(private string imageURL;private string

21、name;public Person(string imageURL,string name)(this.imageURL=imageURL;this.name=name;)DataMember public string ImageURL get return imageURL;set(imageURL=value;)DataMemberpublic string Name(get return name;set(name=value;)服务契约IChat的定义中主要包括Join、Say、Whisper以及Leave四个基本方法。其中 Join表示:进入聊天;Say表示向所有用户广播消息;W

22、hisper表示对指定的用户发送消息;Leave表示离开聊天。需添加ServiceContract属性将IChat接口标记为服务契约,由于要实现客户端与服务端 双向通信的功能,因此还必须设置CallbackContract参数,其参数IChatCallback为Duplex模 式下的客户端回调类型。对于IChat接口中的方法来讲,必须标记OperationContract属性。其中IsOneWay指示在该 消息交换模式下调用方法后是否会立即返回而不需要等待服务端的消息返回;Islnitiating指 示服务方法是否启动一个Session;IsTerminating指示服务方法调用完成是否结束S

23、ession。IChat的定义代码如下:ServiceContract(SessionMode=SessionMode.Required,CallbackContract=typeof(IChatCal Iback)interface IChat(OperationContract(IsOneWay=true,Islnitiating=false,IsTerminating=false)void Say(string msg);OperationContract(IsOneWay=true,Islnitiating=false,IsTerminating=false)void Whisper(

24、string to,string msg);OperationContract(IsOneWay=false,Islnitiating=true,IsTerminating=false)Person Join(Person name);Ope rati onContract(IsOneW ay=true,Islnitiating=false,IsTerminating=true)void Leave();)客户端的回调接口 IChatCallback的定义包括了分别对应Join、Say Whisper以及Leave 的 UserEnter、Receive Receive Whisper 以及

25、UserLeave 四个基本方法。UserEnter 表示当有 新聊天用户加入时所有聊天用户接收到一个相应的通知;Receive表示接收用户广播的消息;ReceiveWhisper表示接收相关用户发来的消息;UserLeave表示当有聊天用户离开时所有聊 天用户接收一个相应的通知。注意在接口定义中,每个服务方法的消息转换模式均设置为One-Way。此外,回调接 口是被本地调用,因此不需要定义Servicecontract属性。回调接口 IChatCallback的定义的代码如下:interface IChatCallback(OperationContract(IsOneWay=true)vo

26、id Receive(Person sender,string message);OperationContract(IsOneWay=true)void ReceiveWhisper(Person sender,string message);OperationContract(IsOneWay=true)void UserEnter(Person person);OperationContract(IsOneWay=true)void U serLeave(Person person);)定义了服务契约以及客户端回调接口之后,仍需要定义一些消息类型用作客户端与服务 之间的交互,具体代码及解

27、释如下:/定义说明消息类型的枚举类型MessageTypepublic enum MessageType Receive,UserEnter,UserLeave,ReceiveWhisper;/ChatEventArgs继承至Event Args,作为传递的消息参数,其包括消息的类型msgType、消息发送者person以及消息的主体内容messagepublic class ChatEventArgs:EventArgs(public MessageType msgType;public Person person;public string message;)最后是对服务契约中IChat接

28、口的实现类ChatService的设计。值得注意的是ChatService 继承了 IChat,不再需要像服务契约IChat 一样添加ServiceContract属性。另外设置了其 ServiceBehavior属性的InstanceContextMode参数来决定实例化方式(PerSession方式或者 PeQall方式)。PerSession表明服务对象的生命周期存活于一个会话期间,在同一个会话期 间对于服务的不同操作的调用都会施加到同一个客户端代理类型的对象上;PeiCall则表示 服务对象是在方法被调用时创建,结束后即被销毁。ServiceBehavior.ConcurrencyMo

29、de参数 的设置则用于控制具体服务对象的并发行为,其包括三种行为:Single:为默认方式。服务实例是单线程的,不接受重入调用(reentrant calls)。也就是说对于 同一个服务实例的多个调用必须排队,直到上一次调用完成后才能继续。Reentrant:和Single 一样,也是单线程的,但能接受重入调用,至于针对同一服务对 象的多个调用仍然需要排队。因为在Single模式下,当方法调用另外一个服务时一,方法会 阻塞,直到所调用的服务完成。如果方法不能重入,那么调用方会因无法接受所调用服务的返回消息,无法解除阻塞状态而陷入死锁。Reentrant模式解决了这种不足。Multiple:允许

30、多个客户端同时调用服务方法。不再有锁的问题,不再提供同步保障。因此使用该模式时,必须自行提供多线程同步机制(比如在本例中给static类型的syncObj 对象加锁)来保证数据成员的读写安全。类ChatService的实现代码及解析如下:ServiceBehavior(InstanceContextMode=InstanceContextMode.PerSession,ConcurrencyMode=ConcurrencyMode.Multiple)public class ChatService:IChat(用于保障多线程同步而设置的对象private static Object syncO

31、bj=new Object();客户端的回调接口对象IChatCallback callback=null;用于广播事件的委托public delegate void ChatEventHandler(object sender,ChatEventArgs e);public static event ChatEventHandler ChatEvent;private ChatEventHandler myEventHandler=null;利用字典对象chatters来保存聊天者对象以及绑定的对应事件委托static DictionaryPerson,ChatEventHandler ch

32、atters=new Dictionary();当前聊天者对象 private Person person;/判断具有相应名字的聊天者是否存在于字典对象中private bool checkIfPersonExists(string name)(foreach(Person p in chatters.Keys)(if(p.Name.Equals(name,StringComparison.OrdinallgnoreCase)(return true;)return false;)/在字典对象中搜索判断其中是否包含了相应名字的聊天者,/如果有则返回其对应的委托ChatEventHandler;

33、否则返回空private ChatEventHandler getPersonHandler(string name)(foreach(Person p in chatters.Keys)(不区分大小写if(p.Name.Equals(name,StringComparison.OrdinallgnoreCase)ChatEventHandler chatTo=null;chatters.TryGetValue(p,out chatTo);return chatTo;)return null;)/在字典对象中搜索判断其中是否包含了相应名字的聊天者,/如果有则返回聊天者对象;否则返回空priva

34、te Person getPerson(string name)(fbreach(Person p in chatters.Keys)(if(p.Name.Equals(name,StringComparison.OrdinallgnoreCase)(return p;)return null;)/加入聊天,如果字典对象chatters中没有同名的聊天者public Person Join(Person person)bool userAdded=false;/创建新的ChatEventHandler类型委托,其指向MyEventHandler。方法 myEventHandler=new Ch

35、atEventHandler(MyEventHandler);执行关键区域,判断是否存在同名聊天者,如果不存在则向字典中添加该对象并将MyEventHandler委托作为其value,以待后面来触发lock(syncObj)(if(!checkIfPersonExists(person.Name)&person!=null)(this.person=person;chatters.Add(person,MyEventHandler);userAdded=true;)如果新的聊天者添加成功,获得一个回调的实例来创建一个消息,并将其广播给其他的聊天者读取字典chatters的所有聊天者对象并返回一

36、个包含了所有对象的列表if(userAdded)callback=OperationContext.Current.GetCallbackChannel();ChatEventArgs e=new ChatEventArgs();e.msgType=MessageType.UserEnter;e.person=this.person;广播有新聊天者加入的消息BroadcastMessage(e);将新加入聊天者对象的委托加到全局的多播委托上ChatEvent+=myEventHandler;Person list=new Personchatters.Count;执行关键区域,将字典chatt

37、ers的所有聊天者对象拷贝至一个聊天者列表上,用于方法返回lock(syncObj)(chatters.Keys.CopyTo(list,0);)return list;)else(return null;)/广播当前聊天者输入的消息给所有聊天者public void Say(string msg)(ChatEventArgs e=new ChatEventArgs();e.msgType=MessageType.Receive;e.person=this.person;e.message=msg;BroadcastMessage(e);)在字典对象chatters中查找具有相应名称的聊天者对

38、象,并异步地触发其对应的ChatEventHandle委托public void Whisper(string to,string msg)(ChatEventArgs e=new ChatEventArgs();e.msgType=MessageType.ReceiveWhisper;e.person=this.person;e.message=msg;try(ChatEventHandler chatterTo;执行关键区域,获取具有相应名称的聊天者对象在chatters中所对应的委托lock(syncObj)chatterTo=getPersonHandler(to);if(chatte

39、rTo=null)(throw new KeyNotFoundException(The person whos name is +to+could not be found);)异步执行委托chatterTo.BeginInvoke(this,e,new AsyncCallback(EndAsync),null);)catch(KeyNotFoundException)()/当聊天者离开时,从字典chatters中移除包含该对象的项。/并从全局的多播委托上移除该对象对应的委托public void Leave()(if(this.person=null)return;获得该聊天者对象所对应的

40、ChatEventHandler委托ChatEventHandler chatterToRemove=getPersonHandler(this.person.Name);执行关键区域,从从字典chatters中移除包含该聊天者对象的项。lock(syncObj)(chatters.Re move(this.person);)从全局的多播委托上移除该对象对应的委托ChatEvent-=chatterToRemove;ChatEventArgs e=new ChatEventArgs();e.msgType=MessageType.UserLeave;e.person=this.person;t

41、his.person=null;将消息广播给其他聊天者BroadcastMessage(e);)/当chatters中的聊天者对象所对应的ChatEventHandler委托被触发时,/MyEventHandler 方法将执行。/该方法通过检查传递过来的ChatEventArgs参数类型来执行相应的客户端的回调接口中的方法。private void MyEventHandler(object sender,ChatEventArgs e)switch(e.msgType)case MessageType.Receive:calIback.Receive(e.person,e.message);

42、break;case MessageType.ReceiveWhisper:callback.ReceiveWhisper(e.person,e.message);break;case MessageType.UserEnter:callback.UserEnter(e.person);break;case MessageType.UserLeave:callback.UserLeave(e.person);break;)catch(Leave();)/异步地触发字典chatters中所有聊天者对象所对应的ChatEventHandler委托。/Beginlnvoke方法可启动异步调用。第一个

43、参数是一个AsyncCallback委托,该委托引用在异步调用完成时要调用的方法。/第二个参数是一个用户定义的对象,该对象可向回调方法传递信息。/Beginlnvoke立即返回,不等待异步调用完成。/Beginlnvoke会返回lAsyncResult,这个结果可用于监视异步调用进度。private void BroadcastMessage(ChatEventArgs e)(ChatEventHandler temp=ChatEvent;if(temp!=null)(fbreach(ChatEventHandler handler in temp.GetInvocationList()(ha

44、ndler.BeginInvoke(this,e,new AsyncCallback(EndAsync),null);)/Endlnvoke方法检索异步调用的结果。/调用Beginlnvoke后可随时调用Endlnvoke方法;如果异步调用尚未完成,Endlnvoke将一直阻止调用线程,直到异步调用完成后才允许调用线程执行。private void EndAsync(IAsyncResult ar)ChatEventHandler d=null;try(/System.Runtime.Remoting.Messaging.AsyncResult 封装了异步委托上的异步操作的结果。/AsyncR

45、esult.AsyncDelegate属性可用于获取在其上调用异步调用的委托对象。System.Runtime.Remoting.Messaging.AsyncResult as res=(System.Runtime.Remoting.Messaging.AsyncResult)ar;d=(ChatEventHandler)asres.AsyncDelegate);d.Endlnvoke(ar);)catch(ChatEvent-=d;)2.宿主端的设计设计好服务之后,宿主端的设计相对简单,只是一个基于控制台的应用程序。首先建立 一个ServiceHost类型的对象host,然后调用Serv

46、iceHost类型的实例方法Open即可启动宿 主监听。需要注意的是在程序末尾应当调用Abort以及Close将资源释放掉,或者利用using 语句来自动释放所占用的资源。下面是宿主端的实现代码以及配置文件内容:static void Main(string args)(Uri uri=new Uri(ConfigurationManager.AppSettingsaddr);ServiceHost host=new ServiceHost(typeof(Chatters.ChatService),uri);host.Open();Console.WriteLine(Chat service

47、Host now listening on Endpoint 0,uri.ToStringO);Console.WriteLine(Press ENTER to stop chat service.);Console.ReadLine();host.Abort();host.Close();)配置文件中,地址为 net.tcp:/localhost:10001/ch at service;绑定为 netTcpB inding,并且 提供可靠性传输、不保障安全性,发送超时时间限定为1秒;契约为Chatters.IChat接口。另外设置了名称为DuplexBinding的绑定配置,其对服务的分流作

48、了限制,并发的Session 个数的上限为lOOOOo 3.客户端的设计客户端的设计主要包括实现客户端代理与配置文件、客户端的回调接口契约的实现以及 客户端登录、主窗体、聊天窗体界面的设计与代码逻辑实现这三部分。表示客户端的各个对 象之间行为关系的序列图如图7所示。图7客户端应用的序列图首先是利用svcutil.exe工具自动生成的客户端代理以及客户端的配置文件,并做出相应 的更改。打开Visual Studio 2005命令提示,转到ChatService所在目录,依次输入以下命令 可以生成同步的客户端代理。svcutil ChatService.exesvcutil*.wsdl*.xsd/

49、language:C#/out:MyProxy.cs/config:app.config输入以下命令可以创建异步的客户端代理:svcutil*.wsdl*.xsd/a/language:C#/out:MyProxy2.cs/config:app.config因为用户加入聊天这个过程是异步执行的,所以可以利用自动生成的异步客户端代理文 件中的BeginJoin与EndJoin方法来替换同步的Join方法。客户端代理的实现代码如下所示:using Common;System.CodeDom.Compiler.GeneratedCodeAttribute(System.ServiceModel,3.

50、0.0.0)System.ServiceModel.ServiceContractAttribute(CallbackContract=typeof(IChatCallback),SessionMode=System.ServiceModel.SessionM ode.Required)public interface IChat(System.ServiceModel.OperationContract Attribute(AsyncPattern=true,Action=http:/tempuri.org/IChat/Join,ReplyAction=http:/tempuri.org/I

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 技术资料 > 其他杂项

本站为文档C TO C交易模式,本站只提供存储空间、用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。本站仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知淘文阁网,我们立即给予删除!客服QQ:136780468 微信:18945177775 电话:18904686070

工信部备案号:黑ICP备15003705号© 2020-2023 www.taowenge.com 淘文阁