利用WCF双工模式实现即时通讯

利用WCF双工模式实现即时通讯,第1张

概述概述 WCF陆陆续续也用过多次,但每次都是浅尝辄止,以将够解决问题为王道,这几天稍闲,特寻了些资料看,昨晚尝试使用WCF的双工模式实现了一个简单的即时通讯程序,通过服务端转发实现客户端之间的通讯。这只

概述 

WCF陆陆续续也用过多次,但每次都是浅尝辄止,以将够解决问题为王道,这几天稍闲,特寻了些资料看,昨晚尝试使用WCF的双工模式实现了一个简单的即时通讯程序,通过服务端转发实现客户端之间的通讯。这只是个Demo,没有考虑异常处理和性能问题。解决方案结构如下:

 

契约

using System;using System.Collections.Generic;using System.linq;using System.ServiceModel;using System.Text;using System.Threading.Tasks;namespace Service.Interface{ [ServiceContract(CallbackContract = typeof(ICallBack))] public interface INoticeOperator { [OperationContract] voID Register(String ID); [OperationContract] voID UnRegister(String ID); [OperationContract] voID SendMessage(String from,String to,String message); }} 

该接口定义了三个行为,分别是:

 •注册
 •注销
 •发消息 

其中,在特性[ServiceContract(CallbackContract = typeof(ICallBack))]中指定了用于服务端回调客户方法的契约ICallBack,其定义如下:

 using System;using System.Collections.Generic;using System.linq;using System.ServiceModel;using System.Text;using System.Threading.Tasks;namespace Service.Interface{ public interface ICallBack { [OperationContract(IsOneWay = true)] voID Notice(String message); }} 

实体 

本Demo只有一个实体,用来表示已经注册用户的ID和对应的回调契约的具体实现的实例:

using Service.Interface;using System;using System.Collections.Generic;using System.linq;using System.Text;using System.Threading.Tasks;namespace Models{ public class ClIEnt { public String ID { get; set; } public ICallBack CallBack { get; set; } }} 

契约的实现代码

 using Models;using Service.Interface;using System;using System.Collections.Generic;using System.linq;using System.ServiceModel;using System.Text;using System.Threading.Tasks;namespace Service{ public class NoticeOperator : INoticeOperator { private static List<ClIEnt> clIEntList = new List<ClIEnt>(); public voID Register(string ID) {  Console.Writeline("register:" + ID);  ICallBack callBack = OperationContext.Current.GetCallbackChannel<ICallBack>();  clIEntList.Add(new ClIEnt() { ID = ID,CallBack = callBack }); } public voID UnRegister(string ID) {  Console.Writeline("unRegister:" + ID);  ClIEnt clIEnt = clIEntList.Find(c => c.ID == ID);  if (clIEnt != null)  {  clIEntList.Remove(clIEnt);  } } public voID SendMessage(string from,string to,string message) {  ClIEnt clIEnt = clIEntList.Find(c => c.ID == to);  if (clIEnt != null)  {  String longMessage = String.Format("message from {0} to {1} at {2} : {3}",from,to,DateTime.Now.ToString("HH:mm:ss"),message);  Console.Writeline(longMessage);  clIEnt.CallBack.Notice(longMessage);  } } }} 

Register方法用来把ClIEnt实体加入到一个列表中,模拟注册行为,Clinet实体包含了用户信息和实现了回调契约的一个实例对象。 

UnRegister方法用来把一个ClIEnt从列表中移除,模拟注销行为。 

SendMessage方法用来发送消息,第一个参数是发送者的ID,第二个参数是消息接受者的ID,第三个参数是发送内容,该方法先将消息在服务端打印出来,然后再回调消息接收者对应的回调契约的具体实现类的实例对象的Notice方法以达到服务端向客户端发送消息的目的。 

宿主

using Service;using Service.Interface;using System;using System.Collections.Generic;using System.linq;using System.ServiceModel;using System.ServiceModel.Description;using System.Text;using System.Threading.Tasks;namespace Hosting{ class Program { static voID Main(string[] args) {  using (ServiceHost host = new ServiceHost(typeof(NoticeOperator)))  {  host.AddServiceEndpoint(typeof(INoticeOperator),new NetTcpBinding(),"net.tcp://127.0.0.1:9527/NoticeOperator");  host.Opened += (s,e) => Console.Writeline("service is running...");  host.open();  Console.Readline();  } } }} 

宿主是一个控制台应用程序,使用的绑定类型为NetTcpBinding,端口是华安的华府的终生代号。 

客户端代码 

实现回调接口

using Service.Interface;using System;using System.Collections.Generic;using System.linq;using System.Text;using System.Threading.Tasks;namespace Test{ class CallBack : ICallBack { public voID Notice(string message) {  Console.Writeline(message); } }} 

模拟注册,发消息和注销

 using Service.Interface;using System;using System.Collections.Generic;using System.linq;using System.ServiceModel;using System.Text;using System.Threading.Tasks;namespace Test{ class Program { static voID Main(string[] args) {  InstanceContext context = new InstanceContext(new CallBack());  using (ChannelFactory<INoticeOperator> factory = new DuplexChannelFactory<INoticeOperator>(context,"net.tcp://127.0.0.1:9527/NoticeOperator"))  {  INoticeOperator proxy = factory.CreateChannel();  String selfID = args[0];  String frIEndID = args[1];  proxy.Register(selfID);  Console.Writeline("----------Register------------");  while(true)  {   String message = Console.Readline();   if (message == "q")   {   proxy.UnRegister(selfID);   break;   }   else   {   proxy.SendMessage(selfID,frIEndID,message);   }  }  } } }} 

在CMD中运行test.exe Joey Ross表示Joey注册,要给他的朋友Ross发送消息;再起一个进程test.exe Ross Joey表示Ross注册,要给他的朋友Joey发送消息。进程启动后输入一些字符按回车即发送至了对方,输入q回车注销并退出程序。如下图所示:

Ross:


Joey:


服务端:

参考资料

 •无废话WCF入门教程五[WCF的通信模式]
 •同事 @麦枫 的代码
 •《WCF全面解析》 

后记 

这仅仅是个Demo,在实际项目中如果同时在线人数非常多,这样做的性能是否可行还需进一步对WCF双工模式的工作方式进行深入学习。 

解决方案下载地址:WCFDemo

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。

总结

以上是内存溢出为你收集整理的利用WCF双工模式实现即时通讯全部内容,希望文章能够帮你解决利用WCF双工模式实现即时通讯所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1257595.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-07
下一篇 2022-06-07

发表评论

登录后才能评论

评论列表(0条)

保存