C#和C++结构体Socket通信

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();


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/406779.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

用 vue-route 的 beforeEach 实现导航守卫(路由跳转前验证登录)

路由跳转前做一些验证&#xff0c;比如登录验证&#xff08;未登录去登录页&#xff09;&#xff0c;是网站中的普遍需求。对此&#xff0c;vue-route 提供的 beforeRouteUpdate 可以方便地实现导航守卫&#xff08;navigation-guards&#xff09;。 导航守卫&#xff08;navig…

统计代码行数

使用终端&#xff0c;cd到项目下 find . "(" -name "*.m" -or -name "*.mm" -or -name "*.cpp" -or -name "*.h" -or -name "*.rss" ")" -print | xargs wc -l 转载于:https://www.cnblogs.com/mo-shou…

基本原理

零序电流保护的基本原理是基于基尔霍夫电流定律&#xff1a;流入电路中任一节点的复电流的代数和等于零&#xff0c;即ΣI0&#xff0c;它是用零序C.T作为取样元件。在线路与电气设备正常的情况下&#xff0c;各相电流的矢量和等于零&#xff08;对零序电流保护假定不考虑不平衡…

GPS数据格式

本菜鸟人&#xff1a;专业导航&#xff0c;善于寻北&#xff0c;精通惯导&#xff0c;兼职GPS&#xff0c;然近日因课题之需&#xff0c;方究机器之视觉也&#xff0c;乃叹人生之无常兮&#xff01;惯导今天不讲&#xff0c;讲的话可以说上6天7夜&#xff08;貌似被拍成电影了&…

C# 的内存拷贝

From: http://www.cnblogs.com/Yjianyong/archive/2010/08/05/1792976.html 近段时间在C#是直接调用动态库比较多&#xff0c;由于有时又需要使用ActiveX控件&#xff0c;往往出现很多的同名的不同命名空间的类&#xff0c;结构等&#xff0c;对不同实体之类的转换是很烦的一件…

也谈如何构建高性能服务端程序

引子&#xff1a;我接触过很多编程语言&#xff0c;接触过各种各样的服务器端开发&#xff0c;Java&#xff0c;Go&#xff0c;Ruby&#xff0c;Javascript等语言&#xff0c;Spring&#xff0c;Node.js&#xff0c;Rails等等常见服务器端框架和编程模型都有接触。这里谈一下我…

Mint-Ui的mt-search点击选中

<template><div class"page-search"><mt-search autofocus v-model"value" placeholder"搜索"> </mt-search><mt-cellv-for"item of filterResult":key"item":title"item"click.nativ…

毕业了!

忆往昔 论文犹如一座大山 日夜码字不知疲倦 写了半天改个没完 良辰美景于我何干 俱往矣 神马检索与期刊 能有几篇是真言 而今迈步从头越 翻过大山尽开颜

网络信息系统(NIS服务器)

分为两部分&#xff1a;一、服务器端配置&#xff1a;1&#xff0e;#检查包是否安装[rootVMLINUX2 shell3]# rpm -qa | grep ypserv ypserv-2.19-32&#xff0e;安装包文件 serv-2.19-3.i386.rpm[rootVMLINUX2 Server]# rpm -ivh ypserv-2.19-3.i386.rpm warning: ypse…

使用C#格式化字符串

From: http://hi.baidu.com/anxing_space/blog/item/6b31186dd6d1d7f942169468.html 今天在博客园上看到一片超赞的C#字符串格式的文章&#xff0c;是我看过最好的一个关于字符格式的文章&#xff0c;转来收藏&#xff0c;顺便向原作者flyingbread表示感谢和尊敬&#xff01; 1…

Tomcat提示Null component

Tomcat提示“严重: Null component Catalina:typeJspMonitor,namejsp,WebModule//localhost/,J2EEApplicationnone,J2EEServernone”错误&#xff0c; 然后无法启动&#xff0c; 这是因为 Tomcat 使用的 jre 版本比应用程序编译的 jre 版本低造成的。 需要把 Tomcat 使用的 jre…

vue如何返回上一页效果

如何通过点击的方式控制当前页返回到上一个路由页面&#xff1a; 1.在当前页面添加返回按钮 <div style"text-align: center"><el-button v-on:click"backHistory">取消</el-button> </div>2.在方法体内写back方法实现点击返回上…

理解 clip

参考 http://developer.51cto.com/art/201009/223326.htm http://vsfor.blog.51cto.com/4165449/1179853 http://www.zhangxinxu.com/wordpress/2011/04/css-cliprect%E7%9F%A9%E5%BD%A2%E5%89%AA%E8%A3%81%E5%8A%9F%E8%83%BD%E5%8F%8A%E4%B8%80%E4%BA%9B%E5%BA%94%E7%94%A8%E4…

机器学习算法基础概念学习总结

转自&#xff1a;http://blog.csdn.net/lantian0802/article/details/38333479 1.基础概念&#xff1a; (1) 10折交叉验证&#xff1a;英文名是10-fold cross-validation&#xff0c;用来测试算法的准确性。是常用的测试方法。将数据集分成10份。轮流将其中的9份作为训练数据&a…