From: http://hi.baidu.com/yangliangwang/blog/item/1a0116138ff098d6f6039ea9.html

最近在用C#做一个项目的时候,Socket发送消息的时候遇到了服务端需要接收C++结构体的二进制数据流,这个时候就需要用C#仿照C++的结构体做出一个结构来,然后将其转换成二进制流进行发送,之后将响应消息的二进制数据流转换成C#结构。

1、仿照C++结构体写出C#的结构来

using System.Runtime.InteropServices;

[Serializable] // 指示可序列化
    [StructLayout(LayoutKind.Sequential, Pack = 1)] // 按1字节对齐
    public struct Operator

{
         public ushort id;
        [MarshalAs(UnmanagedType.ByValArray, SizeConst = 11)] // 声明一个字符数组,大小为11
        public char[] name;
        [MarshalAs(UnmanagedType.ByValArray, SizeConst = 9)]
        public char[] pass;

public Operator(string user, string pass) // 初始化
        {
            this.id = 10000;
            this.name = user.PadRight(11, '\0').ToCharArray();
            this.pass = pass.PadRight(9, '\0').ToCharArray();
        }
    }

2、注意C#与C++数据类型的对应关系

C++与C#的数据类型对应关系表
API数据类型 类型描述 C#类型 API数据类型 类型描述 C#类型
WORD 16位无符号整数 ushort CHAR 字符 char
LONG 32位无符号整数 int DWORDLONG 64位长整数 long
DWORD 32位无符号整数 uint HDC 设备描述表句柄 int
HANDLE 句柄,32位整数 int HGDIOBJ GDI对象句柄 int
UINT 32位无符号整数 uint HINSTANCE 实例句柄 int
BOOL 32位布尔型整数 bool HWM 窗口句柄 int
LPSTR 指向字符的32位指针 string HPARAM 32位消息参数 int
LPCSTR 指向常字符的32位指针 String LPARAM 32位消息参数 int
BYTE 字节 byte WPARAM 32位消息参数 int

整个结构的字节数是22bytes。

对应的C++结构体是:

typedef struct
{
     WORD id;            
    CHAR name[11];
    CHAR password[9];
}Operator;

3、发送的时候先要把结构转换成字节数组

using System.Runtime.InteropServices;

/// <summary>
        /// 将结构转换为字节数组
        /// </summary>
        /// <param name="obj">结构对象</param>
        /// <returns>字节数组</returns>
        public byte[] StructToBytes(object obj)
        {
            //得到结构体的大小
            int size = Marshal.SizeOf(obj);
            //创建byte数组
            byte[] bytes = new byte[size];
            //分配结构体大小的内存空间
            IntPtr structPtr = Marshal.AllocHGlobal(size);
            //将结构体拷到分配好的内存空间
            Marshal.StructureToPtr(obj, structPtr, false);
            //从内存空间拷到byte数组
            Marshal.Copy(structPtr, bytes, 0, size);
            //释放内存空间
            Marshal.FreeHGlobal(structPtr);
            //返回byte数组
            return bytes;

}

接收的时候需要把字节数组转换成结构

/// <summary>
        /// byte数组转结构
        /// </summary>
        /// <param name="bytes">byte数组</param>
        /// <param name="type">结构类型</param>
        /// <returns>转换后的结构</returns>
        public object BytesToStruct(byte[] bytes, Type type)
        {
            //得到结构的大小
            int size = Marshal.SizeOf(type);
            Log(size.ToString(), 1);
            //byte数组长度小于结构的大小
            if (size > bytes.Length)
            {
                //返回空
                return null;
            }
            //分配结构大小的内存空间
            IntPtr structPtr = Marshal.AllocHGlobal(size);
            //将byte数组拷到分配好的内存空间
            Marshal.Copy(bytes, 0, structPtr, size);
            //将内存空间转换为目标结构
            object obj = Marshal.PtrToStructure(structPtr, type);
            //释放内存空间
            Marshal.FreeHGlobal(structPtr);
            //返回结构
            return obj;
        }

4、实际操作:

using System.Collections;
using System.Collections.Generic;
using System.Net;
using System.Net.Sockets;

byte[] Message = StructToBytes(new Operator("user","pass")); // 将结构转换成字节数组

TcpClient socket = new TcpClient();

socket.Connect(ip,port);

NetworkStream ns = Socket.GetStream();

ns.Write(Message,0,Message.Length); // 发送

byte[] Recv = new byte[1024]; // 缓冲

int NumberOfRecv = 0;

IList<byte> newRecv = new List<byte>();
ns.ReadTimeout = 3000;
try
{
do
{
// 接收响应
NumberOfRecv = ns.Read(Recv, 0, Recv.Length);
for (int i = 0; i < NumberOfRecv; i++)
newRecv.Add(Recv[i]);
}
while (ns.DataAvailable);
byte[] resultRecv = new byte[newRecv.Count];
newRecv.CopyTo(resultRecv, 0);

Operator MyOper = new Operator();

MyOper = (Operator)BytesToStruct(resultRecv, MyOper.GetType()); // 将字节数组转换成结构

在这里取值的时候可能会出现只能取到一个字段,剩余的取不到的问题,怎么回事我也搞不懂,反正我的解决办法就是按照字节的顺序从resultRecv里分别取出对应的字段的字节数组,然后解码,例如:

Operator.name是11个字节,最后一位是0,Operator.id是2个字节,那么从第3位到第12位的字节就是Operator.name的内容,取出另存为一个数组MyOperName,Encoding.Default.GetString(MyOperName)就是MyOper.name的内容。

socket.Close();

ns.Close();

C#和C++结构体Socket通信相关推荐

  1. [转]C#和C++结构体Socket通信

    最近在用C#做一个项目的时候,Socket发送消息的时候遇到了服务端需要接收C++结构体的二进制数据流,这个时候就需要用C#仿照C++的结构体做出一个结构来,然后将其转换成二进制流进行发送,之后将响应 ...

  2. c语言send发送结构体,Socket编程中用send发送结构体

    Socket编程中用send发送结构体 原创 2010年04月28日 19:17:00 标签:socket /编程 /struct /google /string /input 11868 最近在开发 ...

  3. C# 倍福ADS带数组成员的结构体数组通信

    C#读写ADS结构体时,如果结构体成员复杂,有数组类型,必须指明数组兼容的c语言类型.有时需要结构体数组缓存这些结构体形式的数据,读取函数与非数组变量不一样. 首先添加以下两个命名空间的引用,ads还 ...

  4. struct sk_buff结构体详解

    struct sk_buff是linux网络系统中的核心结构体,linux网络中的所有数据包的封装以及解封装都是在这个结构体的基础上进行. 1 2 3 4 5 6 7 8 9 10 11 12 13 ...

  5. C# Socket 入门5 UPD 结构体 与 C++ 通信

    这篇文章本来是星期五晚写好了, 因6日去旅游了, 没来得急发上来 1. 同样, 我们先看看这一个比简单的 结构体 代码 using System; using System.Collections.G ...

  6. socket编程之addrinfo结构体与getaddrinfo函数

    1. 概述 IPv4中使用gethostbyname()函数完成主机名到地址解析,这个函数仅仅支持IPv4,且不允许调用者指定所需地址类型的任何信息,返回的结构只包含了用于存储IPv4地址的空间.IP ...

  7. socket传输结构体,c++,发送OK,recv返回字节大小正确但接受数据为空

    socket传输结构体,c++,发送OK,recv返回字节大小正确但接受数据为空 服务端在ubuntu服务器下,客户端在windows下,采用socket进行通信,在客户端接收数据时,出现了诡异的情况 ...

  8. 三、初识Socket套接字结构体

    一.初识Socket套接字结构体 1.通用套接字结构体类型 struct sockaddr{sa_family_t sa_family; //协议簇char sa_data[14]; //协议簇数据} ...

  9. qt socket 传递结构体 结构体中有list_计算机网络应用--Socket编程实验(二)

    本节关于socket部分函数库列表,具体基于socket的实例实现代码见下节.给出基于winSocket的简单websever实例. 2.1 WSAStartup() [函数原型] int WSASt ...

最新文章

  1. win10添加新用户
  2. 把阿里巴巴的核心系统搬到云上,架构上的挑战与演进是什么?
  3. 滴滴升级“极速拼车”:未拼成可享折扣 拼成更便宜
  4. dubbo协议_Dubbo框架支持多少种协议?各有什么特点?文中一一为你揭晓
  5. 前端基础--jquery
  6. 使用 HMAC-SHA1 算法
  7. 【Matlab绘图要点汇总】Matlab图中添加图例+添加轴标签+添加文本+添加注释
  8. 微信与企业微信内嵌浏览器的UserAgent
  9. 使用Python开发小说下载器,不再为下载小说而发愁
  10. 树莓派2B使用360随身WiFi2代连接WiFi
  11. 创建型模式 - 单例模式Singleton
  12. 高德地图-添加自定义图标
  13. php deel views,视图 - Views
  14. mac关闭自动更新提示
  15. 三分建设,七分运营|用现代化安全运营应对数据安全风险
  16. Java项目:精美物流管理系统(java+SpringBoot+Vue+maven+Mysql)
  17. Docker入门详解
  18. libyuv转码出现“corrupted size vs. prev_size”错误
  19. 手把手教你白嫖一个服务器并搭建自己的远程Notebook
  20. Spring框架基础概念(面试概念解答)

热门文章

  1. Bitmap 之 getPixels() 的 stride
  2. 【DICOMDIR专题】DICOMDIR基础知识及常见问题汇总
  3. 资深工程师为何否定这种单例模式
  4. JUnit 4 与 JUnit 3
  5. 图像处理傅里叶变换图像变化_傅里叶变换和图像床单视图。
  6. 算法组合 优化算法_算法交易简化了风险价值和投资组合优化
  7. 平台api对数据收集的影响_收集您的数据不是那么怪异的api
  8. 559. N 叉树的最大深度
  9. 5885. 使每位学生都有座位的最少移动次数
  10. leetcode1328. 破坏回文串