c# socket通信较完善方案

c#的socket通信应用.文件较多.附件为工程. 

core

AbstractBytesWorker.cs    字节工作器(基类),用于用于同一不同功能的字节工作器

BinaryHand.cs  2进制处理器.  ThDispose.cs 处理回收相关

crc  
entity

ThPersonInfo.cs

manager

ThSocketManager.cs  ThSocketManagerBusiness.cs

所有的业务

request

RequestCode.cs  请求码

ThProtocolReq.cs 请求逻辑

ThReqBytesWorker.cs 请求相关的字节工作器

response

respLogic

ThProtocolResp.cs 处理服务器响应的数据.

ThProtocolRespDelegates.cs 所有的代理.用于通知客户的事件.

ThProtocolRespEvents.cs 所有的事件.用于调用客户的.

ThProtocolRespListeners.cs 所有的监听器,用于控制事件如何订阅

ThProtocolRespLogic.cs 处理服务器的数据

ThRespBytesWorker.cs 响应字节处理器

BinaryMessageHandler.cs 处理数据包粘结,包一次数据不足等情况.

ResponseCode.cs 响应码

socket

TAsyncTcpClient.cs tcpClient类,read异步.

testcase

===============================================================

部分类代码:  BinaryMessageHandler

C#代码  

  1. #pragma warning disable 0219
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.IO;
  7. /// <summary>
  8. /// 字节接收处理,粘包问题
  9. /// </summary>
  10. class BinaryMessageHandler : ThDispose
  11. {
  12. List<byte> bytesList = new List<byte>();
  13. private TAsyncTcpClient tcpClient;
  14. public BinaryMessageHandler(TAsyncTcpClient tcpClient)
  15. {
  16. this.tcpClient = tcpClient;
  17. }
  18. public BinaryMessageHandler()
  19. {
  20. }
  21. override public void SelfDispose()
  22. {
  23. tcpClient = null;
  24. bytesList = null;
  25. }
  26. /// <summary>
  27. /// 累积 字节.
  28. /// 每次累积后,测试是否有完整的包.
  29. /// </summary>
  30. /// <param name="buf"></param>
  31. public void Write(byte[] buf)
  32. {
  33. if (buf.Length > 0)
  34. {
  35. //累积字节
  36. bytesList.AddRange(buf);
  37. byte[] bytes = bytesList.ToArray<byte>();
  38. MemoryStream ms = new MemoryStream(bytes);
  39. BinaryReader reader = new BinaryReader(ms);
  40. int header = reader.ReadUInt16();
  41. if (header == ThSocketManager.TH_HEADER)
  42. {
  43. int len = reader.ReadUInt16();
  44. int remainLen = len - 4;
  45. if ((ms.Length - ms.Position) >= remainLen)
  46. {
  47. //有完整的数据包
  48. ms.Position = 0;
  49. byte[] pack = reader.ReadBytes(len);
  50. ReadPackage(pack);
  51. //移除读完的数据包
  52. bytesList.RemoveRange(0, len);
  53. }
  54. }
  55. reader.Close();
  56. ms.Close();
  57. }
  58. }
  59. /// <summary>
  60. /// 读取服务端响应信息.
  61. /// </summary>
  62. /// <param name="bytes"></param>
  63. /// <returns></returns>
  64. public void ReadPackage(byte[] bytes)
  65. {
  66. //处理包头
  67. MemoryStream ms = new MemoryStream(bytes);
  68. ms.Position = 0;
  69. BinaryReader reader = new BinaryReader(ms, Encoding.UTF8);
  70. ushort header = reader.ReadUInt16();
  71. ushort totalLen = reader.ReadUInt16();
  72. ushort respCode = reader.ReadUInt16();
  73. short signature = reader.ReadInt16();
  74. int dataLen = totalLen - ThSocketManager.PREFIX_LENGTH;
  75. byte[] dataBytes = reader.ReadBytes(dataLen);
  76. reader.Close();
  77. ms.Close();
  78. //调用服务端响应,包体处理器.
  79. tcpClient.thProtocolResp.ResponseHandler(respCode, dataBytes);
  80. }
  81. }

BinaryHand

C#代码  

  1. #pragma warning disable 0219
  2. using System.Text;
  3. using System.IO;
  4. class BinaryHand
  5. {
  6. /// <summary>
  7. /// 准备将数据发送至服务端
  8. /// </summary>
  9. /// <param name="clientId"></param>
  10. /// <param name="data"></param>
  11. /// <returns></returns>
  12. public static byte[] ToBytes(ushort requestCode, uint clientId, byte[] dataBytes)
  13. {
  14. MemoryStream ms = new MemoryStream();
  15. BinaryWriter writer = new BinaryWriter(ms);
  16. //2 ushort header
  17. writer.Write(ThSocketManager.TH_HEADER);
  18. //2 ushort total length
  19. ushort packageLen = ThSocketManager.PREFIX_LENGTH;
  20. if (dataBytes != null)
  21. {
  22. packageLen += (ushort)dataBytes.Length;
  23. }
  24. writer.Write(packageLen);
  25. //2 ushort protocol id
  26. writer.Write(requestCode);
  27. //2 short signature
  28. writer.Write((short)0);
  29. //4 unit client id
  30. //writer.Write(clientId);
  31. //x string data
  32. if (dataBytes != null)
  33. writer.Write(dataBytes);
  34. //计算crc,并写入[6,7]位置.
  35. byte[] tmpBytes = ms.ToArray();
  36. short signature = CRC16.Compute(tmpBytes);
  37. long oldPos = ms.Position;
  38. ms.Position = 6;
  39. writer.Write(signature);
  40. ms.Position = oldPos;
  41. //准备输出
  42. byte[] bytes = ms.ToArray();
  43. writer.Close();
  44. ms.Close();
  45. return bytes;
  46. }
  47. public static byte[] ToBytes(RequestCode requestCode, uint clientId, byte[] dataBytes)
  48. {
  49. return ToBytes((ushort)requestCode, clientId, dataBytes);
  50. }
  51. public byte[] ToBytes(uint clientId, string data)
  52. {
  53. byte[] dataBytes = Encoding.UTF8.GetBytes(data);
  54. return ToBytes(RequestCode.None, clientId, dataBytes);
  55. }
  56. /// <summary>
  57. /// 读取服务端响应信息.
  58. /// </summary>
  59. /// <param name="bytes"></param>
  60. /// <returns></returns>
  61. public byte[] FromBytes(byte[] bytes)
  62. {
  63. MemoryStream ms = new MemoryStream(bytes);
  64. ms.Position = 0;
  65. BinaryReader reader = new BinaryReader(ms, Encoding.UTF8);
  66. ushort header = reader.ReadUInt16();
  67. ushort totalLen = reader.ReadUInt16();
  68. ushort protocolId = reader.ReadUInt16();
  69. short signature = reader.ReadInt16();
  70. uint clientId = reader.ReadUInt32();
  71. int dataLen = totalLen - ThSocketManager.PREFIX_LENGTH;
  72. byte[] dataBytes = reader.ReadBytes(dataLen);
  73. reader.Close();
  74. ms.Close();
  75. return dataBytes;
  76. }
  77. }

c# socket通信较完善方案,布布扣,bubuko.com

时间: 2024-10-13 19:01:14

c# socket通信较完善方案的相关文章

C#应用视频教程1.3 Socket通信客户端完善

我们先把前面的代码封装成一个完整的类,因为跟网络相关的方法并不一定是建立socket的服务器和客户端,所以还是应该把两个分开,比如获取本机IP,修改本机IP,PING远程主机这些事情应该放在一个单独的类里面,而且里面的方法应该是静态的(一个应用不会需要多个实例来做上面讲的这些事情),所以我个人建议做成一个单独的类FunctionNetwork,这样可以让主程序很简洁 ? 注意可能涉及到IP地址自动获取和手动修改的方法,需要额外的DLL引用(System.Management),这部分功能暂时用不

使用Socket通信实现Silverlight客户端实时数据的获取(模拟GPS数据,地图实时位置)

原文:使用Socket通信实现Silverlight客户端实时数据的获取(模拟GPS数据,地图实时位置) 在上一篇中说到了Silverlight下的Socket通信,在最后的时候说到本篇将会结合地图.下面就来看看本文实现的功能: Silverlight 与服务器利用Socket通讯,实时从服务器获取数据(本文中的数据是地理坐标),由于没有GPS,所以本文在服务器写了一个构造新坐标的函数(本文是一个三角函数),然后利用Timer组件,实时调用,得到新的坐标,并将新的坐标发送给客户端,客户端接收到发

Java实现简单的socket通信

今天学习了一下java如何实现socket通信,感觉难点反而是在io上,因为java对socket封装已经很完善了. 今天代码花了整个晚上调试,主要原因是io的flush问题和命令行下如何运行具有package的类,不过最后问题基本都解决了,把代码贴出来供大家参考 server public class TcpServer { public static void main(String[] args) throws Exception { ServerSocket server = new S

【转】Winform Socket通信

Socket相关概念[端口] 在Internet上有很多这样的主机,这些主机一般运行了多个服务软件,同时提供几种服务.每种服务都打开一个Socket,并绑定到一个端口上,不同的端口对应于不同的服务(应用程序). 例如:http 使用80端口  ftp使用21端口 smtp 25端口 有两种类型:50000 ?  流式Socket(STREAM): 是一种面向连接的Socket,针对于面向连接的TCP服务应用,安全,但是效率低: ?  数据报式Socket(DATAGRAM): 是一种无连接的So

关于HTTP协议及SOCKET通信

一.HTTP 1.报文结构 HTTP:超文本传输协议,报文分为请求报文和响应报文. 2.端口(tomcat端口) http在熟知的80端口使用TCP的服务:tomcat的默认端口是8080 3.状态码含义 5.get和post方法 get是最简单的一种请求,其主要功能是从服务器端获取用户所需资源,并将其作为响应返回给客户端,这些资源可以是html页面,图片,文档等内容中的任何一种,但需要注意的是,get方法主要用来获取服务器端资源信息,就如同数据库中查询操作一样,不会影响到自身的状态,删除,修改

Linux - 面向连接的socket通信流程

字符串的IP与32的IP的转换 说明 网络上的IP都是数字加点(192.168.0.1)构成 struct in_addr结构使用32位的IP,如 IP(C0A80001)是192.168.0.1 函数原型 int inet_aton(const char *cp, struct in_addr *inp) 将a.b.c.d的IP转换为32位的IP,存储在 inp指针里面 char *inet_ntoa(struct in_addr in) 将32位IP转换为a.b.c.d的格式 说明 a代表

Socket通信——C++服务器端和Java客户端

一句话来说就是,C++和Java 通过socket进行通信.数据传输,通过发送"字节流"即可. 字节对于C++和java来说是通用的,但是传输的过程有许多问题需要注意,我为了弄清楚这个过程,查了一些资料,做了一些整理. 不了解C++ socket编程,可以看这篇博客: Linux 下:socket通信(Linux下,C/C++语言):http://blog.csdn.net/giantpoplar/article/details/47657303 Windows下:winsock:ht

Java与C之间的socket通信

最近正在开发一个基于指纹的音乐检索应用,算法部分已经完成,所以尝试做一个Android App.Android与服务器通信通常采用HTTP通信方式和Socket通信方式.由于对web服务器编程了解较少,而且后台服务器已经采用原始socket实现与c客户端通信,这就要求Android客户端也采用socket实现.所以在开发Android app时采用了原始socket进行编程. 由于算法是用C语言实现的,而Android应用一般是Java开发,这就不可避免得涉及Java和C语言之间的通信问题.一种

Winform Socket通信

原文地址:http://www.cnblogs.com/DrHao/p/4953532.html Socket相关概念[端口] 在Internet上有很多这样的主机,这些主机一般运行了多个服务软件,同时提供几种服务.每种服务都打开一个Socket,并绑定到一个端口上,不同的端口对应于不同的服务(应用程序). 例如:http 使用80端口  ftp使用21端口 smtp 25端口 有两种类型:50000 ?  流式Socket(STREAM): 是一种面向连接的Socket,针对于面向连接的TCP