C#和C++结构体Socket通信(二进制流与结构体)

转载 2013年12月04日 10:33:31

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++数据类型比较及结构体转换

//c++:HANDLE(void *) —- c#:System.IntPtr //c++:Byte(unsigned char) ...
  • Kaizenpku
  • Kaizenpku
  • 2017年04月09日 22:37
  • 817

C#与C++通过socket传送结构体

C#服务端:  1using System;  2using System.Net.Sockets;  3using System.Net;  4using System.IO;  5u...
  • Emilio563
  • Emilio563
  • 2016年10月05日 20:04
  • 1290

如何在C#中直接操作C++结构体

如何在C#中直接操作C++结构体   在C#中调用C++或系统DLL是比较常见的操作。   例如C++中定义的以下结构体: struct RCEStruct {  int Event;     int...
  • starlessnt
  • starlessnt
  • 2008年04月22日 09:24
  • 715

C#中如何定义与C++定义的对应的结构体

1.C++中的结构体 #pragma pack(1) struct A { int a1;//属性a1 int a2;//属性a2 A() { a1= 0; a2= 0; } }...
  • hanhan122655904
  • hanhan122655904
  • 2014年12月22日 16:52
  • 2881

C#调用C++ 平台调用P/Invoke 结构体--结构体嵌套【八】

普通的结构体嵌套很简单,C#中直接定义成对应的结构体即可,这里介绍的是嵌套的结构体以指针的方式表达 【1】嵌套结构体指针 C++代码: typedef struct _testStru10Pr...
  • aoshilang2249
  • aoshilang2249
  • 2014年09月20日 10:06
  • 1442

C#中结构体和类的区别

结构体和类同样能够定义字段,方法和构造函数,都能实例化对象,这样看来结构体和类的功能好像是一样的了,但是他们在数据的存储上是不一样的 C#结构体和类的区别问题:这两种数据类型的本质区别主要是...
  • Circle__Gossoon
  • Circle__Gossoon
  • 2016年08月05日 10:24
  • 1004

[C#]结构体和类的区别

 结构体和类的区别:     在做一个项目时,使用了较多的结构体,并且存在一些结构体的嵌套,即某结构体成员集合包含另一个结构体等,总是出现一些奇怪的错误,才终于下决心好好分析一下到底类和结构体...
  • Andrewniu
  • Andrewniu
  • 2016年08月02日 14:14
  • 855

C# 结构体与类的区别

经常听到有朋友在讨论C#中的结构与类有什么区别.正好这几日闲来无事,自己总结一下,希望大家指点. 1. 首先是语法定义上的区别啦,这个就不用多说了.定义类使用关键字class 定义结构使用关键字...
  • liujunjie612
  • liujunjie612
  • 2017年05月03日 15:02
  • 463

C#中将结构类型数据存储到二进制文件中方法

以往在vb6,v
  • weizhiai12
  • weizhiai12
  • 2014年07月13日 11:49
  • 3731

详解C结构体、C++结构体 和 C++类的区别

先来说说C和C++中结构体的不同 a) C语言中的结构体不能为空,否则会报错 1>d:\myproject\visual studio 2013\projects\myc++\main.c(71): ...
  • Loving_Forever_
  • Loving_Forever_
  • 2016年05月23日 18:38
  • 17225
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C#和C++结构体Socket通信(二进制流与结构体)
举报原因:
原因补充:

(最多只允许输入30个字)