C# Socket发送、接收结构体

 Socket发送:Socket的使用

一、Socket发送结构体

结构体如下:

[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct OutPoint_ST
{
    public int LeftheartX;
    public int LeftHeartY;
    public float WidthHeart;
    public int RightHeartX;
    public int RightHeartY;
    public int ChestX;
    public int ChestY;
    public float WidthChest;
}

结构体已经使用了 [StructLayout(LayoutKind.Sequential, Pack = 0)] 特性,这意味着它的字段按照声明顺序进行内存布局,并且没有填充字节(Pack=0)。这对于通过网络发送结构体数据很有帮助,因为你可以直接读取并发送其字节表示。

但是,请注意以下几点:

  1. Pack = 0 可能会导致某些平台或架构上存在不必要的内存对齐填充。通常,你会希望设置一个合适的打包大小(如 Pack = 1),以消除填充并确保最小的二进制表示。

  2. 对于浮点数 float 类型,不同平台和编译器可能会有不同的字节序(大端序或小端序)。在网络传输中,双方需要约定并保持一致的字节序。

  3. 使用这种方法直接将结构体转换为字节流并发送时,接收方必须知道确切的结构体布局和数据类型,以正确地还原原始数据。

以下是如何将此结构体转换为字节流并发送的示例:

// 假设你有一个OutPoint_ST实例
OutPoint_ST point = new OutPoint_ST() {...};

// 将结构体转换为字节数组
byte[] buffer = System.BitConverter.GetBytes(point.LeftheartX)
    .Concat(System.BitConverter.GetBytes(point.LeftHeartY))
    .Concat(System.BitConverter.GetBytes(point.WidthHeart))
    .Concat(System.BitConverter.GetBytes(point.RightHeartX))
    .Concat(System.BitConverter.GetBytes(point.RightHeartY))
    .Concat(System.BitConverter.GetBytes(point.ChestX))
    .Concat(System.BitConverter.GetBytes(point.ChestY))
    .Concat(System.BitConverter.GetBytes(point.WidthChest))
    .ToArray();

// 发送数据
Socket socket = ... // 获取已连接的Socket实例
socket.Send(buffer);

二、Socket接收结构体

在接收方,需要做相反的操作来解析接收到的字节数组到结构体。

以下是如何在接收端实现这一过程的一个基本示例:

首先,确保接收端已经从Socket接收到完整的结构体字节数组,可以通过循环接收直到达到预期的字节数量。假设你已经将完整结构体的字节数组接收到了byte[] receivedBytes中,并且知道结构体的大小(即所有字段总字节数)。

// 假设你已经接收到了正确的字节数组
byte[] receivedBytes = ... // 从Socket接收的完整结构体字节数组

// 创建一个临时缓冲区来存储结构体
var sizeOfStruct = Marshal.SizeOf<OutPoint_ST>(); // 获取结构体大小
if (receivedBytes.Length < sizeOfStruct)
{
    throw new ArgumentException("Received bytes do not contain a full structure.");
}

OutPoint_ST receivedPoint;

// 将字节数组复制到结构体中
using (var pinnedBuffer = GCHandle.Alloc(receivedBytes, GCHandleType.Pinned))
{
    // 转换为结构体指针
    var pointer = pinnedBuffer.AddrOfPinnedObject();

    // 直接从字节缓冲区读取到结构体中
    receivedPoint = (OutPoint_ST)Marshal.PtrToStructure(pointer, typeof(OutPoint_ST));
}

// 现在receivedPoint包含了从字节流解析出来的结构体数据

上述代码使用了Marshal.PtrToStructure方法来直接从内存中解包字节流到结构体。这种方式假定发送和接收端都在同一平台上运行,且字节序相同。如果两端平台或字节序不一致,需要在转换前调整字节序。

另外,对于大规模的通信应用,依然强烈建议使用序列化库来处理这种转换,以简化工作并处理底层细节,包括但不限于字节序、结构体大小和边界检查等问题。

 

  • 10
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在Python中使用socket发送C结构体需要进行结构体的打包和解包操作,可以使用Python内置的struct模块进行处理。 假设有以下C结构体: ```c struct my_struct { int id; char name[20]; float score; }; ``` 在Python中,可以定义相应的格式字符串来表示这个结构体: ```python import struct # 定义格式字符串 fmt = '<i20sf' # < 表示小端字节序,i 表示 int,20s 表示长度为 20 的字符串,f 表示 float # 打包结构体数据 data = struct.pack(fmt, 1, b'John Doe', 88.5) # 发送数据 s.sendall(data) ``` 在这个例子中,我们使用了格式字符串`'<i20sf'`来表示结构体的布局,其中`<`表示使用小端字节序(与C语言默认的字节序相同),`i`表示一个整数,`20s`表示一个长度为20的字符串,`f`表示一个浮点数。然后使用`struct.pack()`函数将结构体数据打包成二进制字符串,最后使用socket发送数据。 接收端也需要进行相应的解包操作: ```python import struct # 定义格式字符串 fmt = '<i20sf' # 接收数据 data = s.recv(1024) # 解包结构体数据 id, name, score = struct.unpack(fmt, data) name = name.decode('utf-8').rstrip('\x00') # 将字节串转换为字符串,并去掉字符串末尾的空字符 # 输出数据 print('id:', id) print('name:', name) print('score:', score) ``` 在这个例子中,我们使用了`struct.unpack()`函数将二进制数据解包成相应的整数、字符串和浮点数。由于C语言中的字符串是以NULL字符结尾的,因此我们需要使用`rstrip()`函数去掉字符串末尾的空字符。 需要注意的是,C语言中的结构体可能存在字节对齐的问题,因此在Python中打包和解包结构体时需要确保使用相同的字节对齐方式。可以使用`struct.calcsize()`函数来检查格式字符串的大小是否与C语言中的结构体大小相同。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wangnaisheng

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值