14.7 Socket 循环结构体传输
在上述内容中笔者通过一个简单的案例给大家介绍了在套接字编程中如何传递结构体数据,本章将继续延申结构体传输,在某些时候例如我们需要传输一些当前系统的进程列表信息,或者是当前主机中的目录文件,此时就需要使用循环结构体传输功能,循环传输结构体的关键点在于,客户端发送结构体数据之前需要通过一次通信来告诉服务端需要接收的次数,当服务端接收到次数时则可利用接收计数器依次循环接收数据直到客户端完整所有数据包的发送。
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 循环结构体传输的更多相关文章
- C#与C++通过socket传送结构体
C#服务端: using System; using System.Net.Sockets; using System.Net; using System.IO; using System.Diagn ...
- Bash脚本编程学习笔记07:循环结构体
本篇中涉及到算术运算,使用了$[]这种我未在官方手册中见到的用法,但是确实可用的,在此前的博文<Bash脚本编程学习笔记03:算术运算>中我有说明不要使用,不过自己忘记了.大家还是尽量使用 ...
- socket发送结构体
struct send_info {char info_from[20]; //发送者IDchar info_to[20]; //接收者IDint info_length; //发送的消息主体的长度c ...
- 2. socket结构体——表示socket地址
一.两种通用socket结构体 1. sockaddr struct sockaddr { sa_family_t sa_family; // 地址族 char sa_data[14]; // 存放s ...
- C# Socket 入门4 UPD 发送结构体(转)
今天我们来学 socket 发送结构体 1. 先看要发送的结构体 using System; using System.Collections.Generic; using System.Text; ...
- 全国计算机等级考试二级教程-C语言程序设计_第14章_结构体、共用体和用户定义类型
函数的返回值是结构体类型 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> struct ...
- IPv4地址结构体sockaddr_in详解
sockaddr_in结构体定义 struct sockaddr_in { sa_family_t sin_family; //地址族(Address Family) uint16_t sin_por ...
- C++ 结构体+数组+取随机数 案例(打印3名老师 带着 5名学生)结构体
1 //结构体案列 2 3 #include<iostream> 4 #include<string> 5 #include<ctime> 6 using name ...
- C与C# socket 跨平台通讯传输结构体
最近需要写一个C组成的服务器端与C#的客户端进行交互的软件,刚开始写的时候发现C#端解析时候出现了故障,经过仔细研究后发现原因是发送方传输太快,出现了所谓粘包的现象.也就是在C#端的Receive() ...
- struct socket结构体详解
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://weiguozhihui.blog.51cto.com/3060615/15852 ...
随机推荐
- Java Socket Demo
服务端: package com.sux.demo; import java.io.*; import java.net.ServerSocket; import java.net.Socket; i ...
- C++岗位面试真题宝典 -- 语言基础篇
一.C++语言基础 1.1.1 简述下C++语言的特点 参考回答: C++在C语言基础上引入了面对对象的机制,同时也兼容C语言. C++有三大特性(1)封装.(2)继承.(3)多态: C++语言编写出 ...
- 小白学标准库之 log
日常开发中,日志 log 几乎是必不可少.本文旨在介绍 log 的使用和内部实现等. 1. log 使用及实现 package main import ( "fmt" " ...
- STM32CubeMX教程22 FSMC - 8080并行接口TFT-LCD驱动
1.准备材料 开发板(正点原子stm32f407探索者开发板V2.4) STM32CubeMX软件(Version 6.10.0) 野火DAP仿真器 keil µVision5 IDE(MDK-Arm ...
- Pickle反序列化学习
什么是Pickle? 很简单,就是一个python的序列化模块,方便对象的传输与存储.但是pickle的灵活度很高,可以通过对opcode的编写来实现代码执行的效果,由此引发一系列的安全问题 Pick ...
- Linux-文件指令-cat-less-head-touch
- [转帖]002、体系结构之TiDB Server
TiDB Server 1.TiDB总览 1.1.TiDB Server架构 1.2.TiDB Server 主要功能: 2.SQL语句处理 语句的解析和编译 SQL层 协议层 上下文 解析层 逻辑优 ...
- [转帖]Windows平台下使用 Rclone 挂载 OneDrive 为本地硬盘
https://zhuanlan.zhihu.com/p/139200172 Rclone (rsync for cloud storage) 是一个命令行程序,用于同步文件和目录,支持常见的 Ama ...
- [转帖]Kafka 核心技术与实战学习笔记(六)kafka线上集群部署方案
一.操作系统-Linux Kafka是JVM系的大数据框架 kafka由Scala语言和Java语言编写而成,编译之后的源代码就是普通的".class"文件 使用Linux kaf ...
- [转帖]Sosreport:收集系统日志和诊断信息的工具
https://zhuanlan.zhihu.com/p/39259107 如果你是 RHEL 管理员,你可能肯定听说过 Sosreport :一个可扩展.可移植的支持数据收集工具.它是一个从类 Un ...