在上述内容中笔者通过一个简单的案例给大家介绍了在套接字编程中如何传递结构体数据,本章将继续延申结构体传输,在某些时候例如我们需要传输一些当前系统的进程列表信息,或者是当前主机中的目录文件,此时就需要使用循环结构体传输功能,循环传输结构体的关键点在于,客户端发送结构体数据之前需要通过一次通信来告诉服务端需要接收的次数,当服务端接收到次数时则可利用接收计数器依次循环接收数据直到客户端完整所有数据包的发送。

14.7.1 服务端实现

多条结构体的传输方式与单条从原理上一致,只是多条结构体在传输时需要提前告知服务端我需要分几次将结构体传输给对方,因为数据包最大单次可发送8192字节,所以如果结构过多则需要分批次进行传输,如下是服务端实现代码片段,在代码中首先我们接收客户端发来的循环次数,该次数是一个字符串类型的,为了能用于循环体内,需要通过atoi(count)将其转换为一个整数,接着就是在循环体内不断地调用recv函数接收数据包,直到循环结束为止。

#include <iostream>
#include <winsock2.h> #pragma comment(lib,"ws2_32.lib") typedef struct
{
char HostName[32];
char Buffer[32];
}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; msgsock = accept(sock, (LPSOCKADDR)0, (int*)0);
if (msgsock != INVALID_SOCKET)
{
// 接收循环次数
char count[32] = { 0 };
int recv_count_flag = recv(msgsock, count, sizeof(count), 0);
if (recv_count_flag != 0)
{
// 得到需要循环接收的次数
int index = atoi(count);
std::cout << "总共循环接收: " << count << " 次" << std::endl; for (int x = 0; x < index; x++)
{
char recv_buf[4096] = { 0 }; // 循环输出接收结果
int recv_flag = recv(msgsock, recv_buf, sizeof(recv_buf), 0);
if (recv_flag != 0)
{
// 接收到结构,强制类型转换
message* msg = (message*)recv_buf; std::cout << "用户名: " << msg->HostName << "数据: " << msg->Buffer << std::endl; // 发送成功标志
send(msgsock, "success", 7, 0);
}
}
}
} closesocket(sock);
WSACleanup();
return 0;
}

14.7.2 客户端实现

相对于服务端而言,客户端首先需要准备好一个待发送结构体链表,此处通过使用vector<message>的方式接收结构体链表,并通过sprintf()函数将循环次数由整数格式化为字符串,并将次数发送给服务端,当服务端接收到发送次数后会等待客户端向其发送对应数量的结构体,此时客户端只需要send循环发送即可。

#include <iostream>
#include <vector>
#include <winsock2.h> #pragma comment(lib,"ws2_32.lib") using namespace std; typedef struct
{
char HostName[32];
char Buffer[32];
}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;
} // 模拟元素填充
std::vector<message> vect; for (int x = 0; x < 10; x++)
{
message ptr;
// 填充参数
sprintf(ptr.HostName, "lyshark %d", x);
sprintf(ptr.Buffer, "hello lyshark %d", x);
vect.push_back(ptr);
} // 发送循环次数
char count[32] = { 0 }; // 整数转为字符串
sprintf(count, "%d", vect.size()); int send_count_flag = send(sock, count, sizeof(count), 0); if (send_count_flag != 0)
{
std::cout << "发送循环次数: " << count << std::endl; // 循环发送数据
for (int x = 0; x < vect.size(); x++)
{
char send_buf[4096] = { 0 }; // 发送字节序
memcpy(send_buf, &vect[x], sizeof(message));
int send_flag = send(sock, send_buf, sizeof(send_buf), 0);
if (send_flag != 0)
{
char recv_buf[32] = { 0 };
recv(sock, recv_buf, sizeof(recv_buf), 0);
std::cout << "发送完成,接收状态码: " << recv_buf << std::endl;
}
}
} closesocket(sock);
WSACleanup();
return 0;
}

至此读者可分别编译并运行服务端与客户端,此时会看到如下图所示的结构体输出;

14.7 Socket 循环结构体传输的更多相关文章

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

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

  2. Bash脚本编程学习笔记07:循环结构体

    本篇中涉及到算术运算,使用了$[]这种我未在官方手册中见到的用法,但是确实可用的,在此前的博文<Bash脚本编程学习笔记03:算术运算>中我有说明不要使用,不过自己忘记了.大家还是尽量使用 ...

  3. socket发送结构体

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

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

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

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

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

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

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

  7. IPv4地址结构体sockaddr_in详解

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

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

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

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

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

  10. struct socket结构体详解

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

随机推荐

  1. Linux day3:⽹络不通排查流程 linux重要数据文件 系统优化相关 上传下载 文件权限 所属用户及用户组

    目录 ⽹络不通排查流程 linux重要数据文件 etc⽬录下重要的数据⽂件 usr⽬录下重要的数据⽂件 var⽬录下重要的数据⽂件 proc⽬录重要的数据⽂件 系统优化相关 环境变量 下载软件优化操作 ...

  2. Safari 14.0 的功臣 Webp?

    俗话说:一图胜千言.在网上,图片虽然可以让用户更加简单明了地看到更多信息,但是图片体积也可以抵过上千字节甚至更多.研究表明,打开一个 HTTP 网页,其中图片平均占比为 64%.在图片占比如此高的情况 ...

  3. Mongodb--用户/权限

    mongodb分为管理员用户和普通用户,并且还有个验证库,建立用户时use到的库(验证库),信息就存储在admin数据库下. 在使用用户时,要加上验证库才能登录,对于管理员用户,必须在admin下创建 ...

  4. C# 32位程序申请大内存

    VS2015 项目-->属性-->生成事件 在"后期生成事件命令行"中输入: cd /d $(DevEnvDir) cd.. cd.. cd VC\bin editbi ...

  5. 6.0 《数据库系统概论》之关系数据库的规范化理论(数据依赖对表的影响[插入-删除-修改-冗余]、1NF-2NF-3NF-BCNF-4NF、函数依赖与多值依赖)

    前言 本篇文章学习书籍:<数据库系统概论>第5版 王珊 萨师煊编著 视频资源来自:数据库系统概论完整版(基础篇+高级篇+新技术篇) 由于 BitHachi 学长已经系统的整理过本书了,我在 ...

  6. SCA技术进阶系列(一):SBOM应用实践初探

    现代软件都是组装的而非纯自研.随着开源组件在数字化应用中的使用比例越来越高,混源开发已成为当前业内主流开发方式.开源组件的引入虽然加快了软件开发效率,但同时将开源安全问题引入了整个软件供应链.软件组成 ...

  7. <vue 基础知识 5、事件监听>

    代码结构 一.     v-on基本使用 1.效果 按钮点击一下数字增加1 2.代码 01-v-on基本使用.html <!DOCTYPE html> <html lang=&quo ...

  8. freeswitch的任务引擎问题与解决方案

    概述 freeswitch核心框架中有一个定时任务系统task,在开发过程中用来做一些延时操作和异步操作很方便. 我们在VOIP的呼叫流程中,经常会有一些对实时性要求没那么高的操作,或者会有阻塞流程的 ...

  9. 如何将接口的返回值中所需信息提取出来作为其他接口的入参使用(postman与jmeter的使用)

    一.背景: 偶尔会用到一个场景,两个接口之前的调用有依赖关系,将其中一个的返回参数中的部分信息取出来作为入参在第二个接口中使用,代码内是比较好实现,只要定义一个变量,用于参数传递. 如果是测试过程中使 ...

  10. 【Freertos】任务切换分析

    任务切换实现 xPortPendSVHandler: mrs r0, psp // 获取进入异常时的进程栈 isb ldr r3, =pxCurrentTCB // 加载线程控制块地址到r3 ldr ...