科技行者

行者学院 转型私董会 科技行者专题报道 网红大战科技行者

知识库

知识库 安全导航

至顶网网络频道轻松定义自己的网络通讯协议

轻松定义自己的网络通讯协议

  • 扫一扫
    分享文章到微信

  • 扫一扫
    关注官方公众号
    至顶头条

每次编写设计网络通讯程序时,总面对一个问题,就是要自定义一组应用协议,然后再写相应的方法来解析协议,并提供相应的接口供上层调用。

作者:中国IT实验室 2007年9月16日

关键字: 路由协议 EIGRP igrp SNMP CISCO 网络协议

  • 评论
  • 分享微博
  • 分享邮件

  每次编写设计网络通讯程序时,总面对一个问题,就是要自定义一组应用协议(即通讯协议),然后再写相应的方法来解析协议,并提供相应的接口供上层调用。如果只是简单的文本信息通讯还容易,但要交换一些控制信息或结构复杂的数据时,比如做联机游戏,更是让人头疼。

  

  最近突然想到一个点子,可以用对象串行化技术将对象直接转换为二进制数据发送,然后接收时直接还原为对象。具体过程是,将要发送的数据放在一个HashTable中,串行化后发送出去,在接收方接收到数据并还原为HashTable,根据预先约定好的Key和获取自己关心的数据。在这种情况下,定义通讯协议的内容实质上也就只是指定一组Key就行了。再也不用做那些规定第几个字段是什么类型有多长的烦躁的事情了。

  

  可能很多人很善于用XML,也将之广泛用于网络通讯。XML有不可比拟的好处,因为它是同平台无关的,而且基本上任何开发语言都有现成库来解析XML。这个和我的观点并不冲突。对象串行化并不局限于二进制数据。C#里有丰富的方法,可以将对象串行化为XML文档,也支持用SOAP协议来串行化数据。所以只要用公共的串行化标准来串行化对象,也可以达到跨平台、跨语言的目的。其实现在流行的Web Service其核心技术也就大概是这样吧。

  

  原理说完了,贴段代码做个例子。ObjectTransferClient(简称OTC)是一个利用UDP协议及二进制对象串行化的包括对象发送和接收的库。调用方法很简单,用Send发送对象,响应ReceiveObject事件来处理接收的对象。至于具体细节就不多叙述了,相信有一定C#基础的人能轻松看懂的。

  

  这一原理的应用潜力是巨大的,我在这里抛砖引玉,还请指教。

  using System;

  using System.Net.Sockets;

  using System.Net;

  using System.Runtime.Serialization.Formatters.Binary;

  using System.Threading;

  namespace OTC

  {

  ///

  /// 对象传送器,使用UDP协议通过网路在不同主机间传送对象。

  ///

  public class ObjectTransferClient : IDisposable

  {

  private Thread ListenThread;

  private BinaryFormatter Serializer = new BinaryFormatter();

  private int m_Port;

  private UdpClient m_Client;

  private bool m_IsStart;

  private bool m_IsConnected;

  

  ///

  /// 接收到对象事件

  ///

  public event ReceiveObjectEventHandler ReceiveObject;

  

  ///

  /// 构建一个对象传送器,默认端口7321

  ///

  public ObjectTransferClient() : this(7321)

  {

  //

  // TODO: 在此处添加构造函数逻辑

  //

  }

  

  ///

  /// 指定端口号构建一个对象传送器。

  ///

  /// 端口号

  public ObjectTransferClient(int port)

  {

  this.m_Port = port;

  this.m_IsConnected = false;

  this.m_IsStart = false;

  }

  

  ///

  /// 初始化传送器并开始工作

  ///

  public void Start()

  {

  if (!this.m_IsStart)

  {

  this.m_Client = new UdpClient(this.m_Port);

  ListenThread = new Thread(new ThreadStart(this.Listen));

  ListenThread.Start();

  this.m_IsStart = true;

  }

  }

  

  ///

  /// 使用指定的主机名和端口连接默认的远程主机

  ///

  /// 主机名

  /// 端口

  public void Connect(string hostname, int port)

  {

  this.m_Client.Connect(hostname, port);

  }

  

  ///

  /// 使用指定的IP地址和端口连接默认的远程主机

  ///

  /// IP地址

  /// 端口

  public void Connect(IPAddress ipaddress, int port)

  {

  this.m_Client.Connect(ipaddress, port);

  }

  

  ///

  /// 使用网络终结点连接默认的远程主机

  ///

  /// 网络端点

  public void Connect(IPEndPoint iep)

  {

  this.m_Client.Connect(iep);

  }

  

  private byte[] CreateArgPackage(object obj)

  {

  IPEndPoint local = new IPEndPoint(IPAddress.Any, this.m_Port);

  System.IO.MemoryStream outStream = new System.IO.MemoryStream();

  this.Serializer.Serialize(outStream, new ReceiveObjectEventArgs(obj, local));

  return outStream.ToArray();

  }

  

  ///

  /// 将对象发送到默认主机。调用此方法前必须先调用Connect方法连接默认主机。

  ///

  /// 要发送的对象

  public void Send(object obj)

  {

  if (this.IsConnected)

  {

  byte[] data = this.CreateArgPackage(obj);

  this.m_Client.Send(data, data.Length);

  }

  else

  {

  throw new Exception("必须先连接默认主机");

  }

  }

  

  ///

  /// 将对象发送到指定的主机。若调用了Connect方法连接了默认主机,则此方法不可用。

  ///

  /// 要发送的对象

  /// 目标主机的网络端点

  public void Send(object obj, IPEndPoint remoteIEP)

  {

  if (this.IsConnected)

  {

  throw new Exception("已经连接了默认主机");

  }

  else

  {

  byte[] data = this.CreateArgPackage(obj);

  this.m_Client.Send(data, data.Length, remoteIEP);

  }

  }

  

  ///

  /// 监听接收数据线程方法

  ///

  protected void Listen()

  {

  BinaryFormatter Serializer = new BinaryFormatter();

  IPEndPoint RemoteIpEndPoint = new IPEndPoint(IPAddress.Any, 0);

  while (true)

  {

  try

  {

  object revobj = Serializer.Deserialize(new System.IO.MemoryStream(m_Client.Receive(ref RemoteIpEndPoint)));

  ReceiveObjectEventArgs revarg = (ReceiveObjectEventArgs)revobj;

  RemoteIpEndPoint.Port = revarg.RemoteIEP.Port;

  revarg.RemoteIEP = RemoteIpEndPoint;

  if (this.ReceiveObject != null)

  {

  this.ReceiveObject(this, revarg);

  }

  }

  catch

  {

  break;

  }

  }

  }

  

  #region 公共属性区

  

  ///

  /// 返回或设置接收对象的端口号

  ///

  public int Port

  {

  get

  {

  return this.m_Port;

  }

  set

  {

  this.m_Port = value;

  }

  }

  

  ///

  /// 返回对象发送器是否已经初始化并开始工作

  ///

  public bool IsStart

  {

  get

  {

  return this.m_IsStart;

  }

  }

  

  ///

  /// 返回对象发送器是否已经连接默认远程主机

  ///

  public bool IsConnected

  {

  get

  {

  return this.m_IsConnected;

  }

  }

  

  #endregion

  

  #region IDisposable 成员

  

  public void Dispose()

  {

  // TODO: 添加 ObjectTransferClient.Dispose 实现

  this.m_Client.Close();

  this.ListenThread.Abort();

  }

  #endregion

  }

  

  ///

  /// 接收对象事件参数

  ///

  [Serializable]

  public class ReceiveObjectEventArgs : EventArgs

  {

  private object _obj;

  private System.Net.IPEndPoint _iep;

  

  ///

  /// 构建一个接收对象事件的参数

  ///

  /// 接收到的对象

  /// 发送者的网络端点

  internal ReceiveObjectEventArgs(object obj, System.Net.IPEndPoint iep)

  {

  this._obj = obj;

  this._iep = iep;

  }

  

  ///

  /// 构建一个空的接收对象事件参数

  ///

  public ReceiveObjectEventArgs():this(null, null)

  {

  }

  

  ///

  /// 接收到的对象

  ///

  public object Obj

  {

  get

  {

  return this._ob

    • 评论
    • 分享微博
    • 分享邮件
    邮件订阅

    如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。

    重磅专题
    往期文章
    最新文章