当在套接字编程中传输结构体时,可以将结构体序列化为字符串(即把结构体的所有成员打包成一个字符串),然后将字符串通过套接字传输到对端,接收方可以将字符串解析为结构体,然后使用其中的成员数据。这种方法通常被称为序列化(Serialization)和反序列化(Deserialization),本章中我们可以采用将一个结构体序列化为一个纯字符串,然后将该字符串通过套接字传输给对端,当对端收到后只需要将字节序强制转换为对等的结构体指针即可实现对该结构的解析。

14.6.1 服务端实现

首先来看服务端的实现流程,笔者定义了message结构体变量,代码中在接收到套接字传输过来的字符串之后,通过(message*)recv_buf的方式将该套接字强制转换为一个结构体指针,并赋值给message* msg指针,此时在服务端则可通过->的方式输出当前结构体内的完整成员参数。

#include <iostream>
#include <winsock2.h> #pragma comment(lib,"ws2_32.lib") // 定义一个自定义结构体
typedef struct
{
unsigned short uuid;
char HostName[32];
char Buffer[32]; struct
{
char HostAddress[1024];
char HostPassword[1024];
char HostPort[1024];
}HostInfo; unsigned short cmd_type;
}message; int main(int argc, char* argv[])
{
WSADATA WSAData;
SOCKET sock; WSAStartup(MAKEWORD(2, 0), &WSAData);
sock = socket(AF_INET, SOCK_STREAM, 0);
if (sock == INVALID_SOCKET)
{
std::cout << "创建套接字失败" << std::endl;
} struct sockaddr_in ServerAddr;
ServerAddr.sin_family = AF_INET;
ServerAddr.sin_port = htons(9999);
ServerAddr.sin_addr.s_addr = INADDR_ANY; // 绑定套接字
auto res = bind(sock, (LPSOCKADDR)&ServerAddr, sizeof(ServerAddr));
if (res == SOCKET_ERROR)
{
std::cout << "绑定失败" << std::endl;
} // 侦听套接字
res = listen(sock, 10);
if (res == SOCKET_ERROR)
{
std::cout << "侦听失败" << std::endl;
} SOCKET msgsock; // 用于接收结构
char recv_buf[4096] = { 0 }; msgsock = accept(sock, (LPSOCKADDR)0, (int*)0);
if (msgsock != INVALID_SOCKET)
{
recv(msgsock, recv_buf, sizeof(recv_buf), 0); // 接收到结构,强制类型转换
message* msg = (message*)recv_buf; std::cout << "ID: " << msg->uuid << " 主机名: " << msg->HostName
<< " 数据包: " << msg->Buffer << " 主机地址: " << msg->HostInfo.HostAddress
<< "主机密码: " << msg->HostInfo.HostPassword << std::endl;
} closesocket(sock);
WSACleanup();
return 0;
}

14.6.2 客户端实现

对于客户端来说,在定义好全局message结构体之后直接通过字符串拷贝方法实现对全局结构的填充,当全局结构体被填充后直接使用memcpy(send_buf, &msg, sizeof(message))将该结构体的字节拷贝到send_buf这个缓冲区内,最后调用send()发送这段缓冲区即可实现。

#include <iostream>
#include <winsock2.h> #pragma comment(lib,"ws2_32.lib") // 定义一个自定义结构体
typedef struct
{
unsigned short uuid;
char HostName[32];
char Buffer[32]; struct
{
char HostAddress[1024];
char HostPassword[1024];
char HostPort[1024];
}HostInfo; unsigned short cmd_type;
}message; // 定义变量
message msg; int main(int argc, char* argv[])
{
WSADATA WSAData;
SOCKET sock; WSAStartup(MAKEWORD(2, 0), &WSAData);
sock = socket(AF_INET, SOCK_STREAM, 0);
if (sock == INVALID_SOCKET)
{
std::cout << "创建套接字失败" << std::endl;
} struct sockaddr_in ClientAddr;
ClientAddr.sin_family = AF_INET;
ClientAddr.sin_port = htons(9999);
ClientAddr.sin_addr.s_addr = inet_addr("127.0.0.1"); // 链接对端
auto res = connect(sock, (LPSOCKADDR)&ClientAddr, sizeof(ClientAddr));
if (res == SOCKET_ERROR)
{
std::cout << "链接失败." << std::endl;
} // 首先需要定义一个变量
char send_buf[4096] = { 0 }; // 拷贝外部变量
msg.uuid = 1001;
strcpy(msg.HostName, "lyshark PC");
strcpy(msg.Buffer, "hello world"); // 拷贝内部HostInfo
strcpy(msg.HostInfo.HostAddress, "192.168.1.1");
strcpy(msg.HostInfo.HostPassword, "12345678");
strcpy(msg.HostInfo.HostPort, "8888"); // 拷贝内部Type
msg.cmd_type = 1; // 发送字节序
memcpy(send_buf, &msg, sizeof(message));
send(sock, send_buf, sizeof(send_buf), 0); closesocket(sock);
WSACleanup();
return 0;
}

运行上述代码片段,读者可看到结构体已被正确的传输给服务端,效果图如下所示;

14.6 Socket 应用结构体传输的更多相关文章

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

    C#服务端: using System; using System.Net.Sockets; using System.Net; using System.IO; using System.Diagn ...

  2. socket发送结构体

    struct send_info {char info_from[20]; //发送者IDchar info_to[20]; //接收者IDint info_length; //发送的消息主体的长度c ...

  3. 2. socket结构体——表示socket地址

    一.两种通用socket结构体 1. sockaddr struct sockaddr { sa_family_t sa_family; // 地址族 char sa_data[14]; // 存放s ...

  4. C# Socket 入门4 UPD 发送结构体(转)

    今天我们来学 socket  发送结构体 1. 先看要发送的结构体 using System; using System.Collections.Generic; using System.Text; ...

  5. 全国计算机等级考试二级教程-C语言程序设计_第14章_结构体、共用体和用户定义类型

    函数的返回值是结构体类型 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> struct ...

  6. IPv4地址结构体sockaddr_in详解

    sockaddr_in结构体定义 struct sockaddr_in { sa_family_t sin_family; //地址族(Address Family) uint16_t sin_por ...

  7. C++ 结构体+数组+取随机数 案例(打印3名老师 带着 5名学生)结构体

    1 //结构体案列 2 3 #include<iostream> 4 #include<string> 5 #include<ctime> 6 using name ...

  8. C与C# socket 跨平台通讯传输结构体

    最近需要写一个C组成的服务器端与C#的客户端进行交互的软件,刚开始写的时候发现C#端解析时候出现了故障,经过仔细研究后发现原因是发送方传输太快,出现了所谓粘包的现象.也就是在C#端的Receive() ...

  9. struct socket结构体详解

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://weiguozhihui.blog.51cto.com/3060615/15852 ...

  10. socket编程相关的结构体和字节序转换、IP、PORT转换函数

    注意:结构体之间不能直接进行强制转换, 必须先转换成指针类型才可以进行结构体间的类型转换, 这里需要明确的定义就是什么才叫强制转换. 强制转换是将内存中一段代码以另一种不同类型的方式进行解读, 因此转 ...

随机推荐

  1. c#-微软2

    练习-编写第一个代码: 在第一次练习中你将使用c#将神圣的程序员用语打印到控制台的标准输出 编写第一行代码: 在软件开发者中,有这么一个传统,那就是将"Hello World!"这 ...

  2. VL02N、VL09交货单相关增强

    一.业务需求 当前台操作过账发货或冲销时,需要将数据实时同步到第三方系统,因此需要开发增强 在用户出口MV50AFZ1->userexit_save_document中加入接口逻辑即可 定期更文 ...

  3. WCF 使用动态代理精简代码架构 (WCF动态调用)

    使用Castle.Core.dll实现,核心代码是使用Castle.DynamicProxy.ProxyGenerator类的CreateInterfaceProxyWithoutTarget方法动态 ...

  4. Step by step guide to becoming a C++ developer in 2023

    https://roadmap.sh/cpp https://roadmap.sh/backend

  5. mysql8.0环境搭建linux

    本文主要介绍如何在linux环境(64位)下搭建mysql8.0的数据库环境 1.到指定目录下下载安装包 [root@minio3 ~]# cd /usr/local/src [root@minio3 ...

  6. Proxifier 2023年11月时最新版 激活教程

    前言 Proxifier 是一款功能非常强大的socks5客户端,可以让不支持通过代理服务器工作的网络程序能通过HTTPS或SOCKS代理或代理链.支持64位系统支持Xp,Vista,Win7,支持s ...

  7. java进阶(30)--Hashtable集合与Properties集合

    一.Hashtable简介 1.HashMap与Hashtable区别 Hashtable的key与value均不能为空,而HashMap均可以 2.Hashtable方法带有Synchronized ...

  8. 机器学习-线性分类-支持向量机SVM-SMO算法-14

    目录 1. SVM算法总结 2. SMO算法 1. SVM算法总结 选择 核函数 以及对应的 超参数 为什么要选择核函数? 升维 将线性问题不可分问题 升维后转化成 线性可分的问题 核函数 有那些? ...

  9. 小白学标准库之 log

    日常开发中,日志 log 几乎是必不可少.本文旨在介绍 log 的使用和内部实现等. 1. log 使用及实现 package main import ( "fmt" " ...

  10. FGC频繁导致CPU 飙升定位及JVM配置优化总结

    本文为博主原创,未经允许不得转载: 目录: 1. 定位消耗cpu 的服务进程和线程 2. 定位FGC 的原因 3. 定位jvm 参数是否导致FGC 4. 调试最优解的 jvm 配置 描述:项目中存在一 ...