当前位置: 首页 > article >正文

从“三次握手”到文件落地:用Wireshark抓包带你彻底搞懂C++ Socket文件传输全过程

从“三次握手”到文件落地用Wireshark抓包带你彻底搞懂C Socket文件传输全过程当你在浏览器下载文件时是否好奇过数据是如何跨越网络准确无误地到达你的电脑本文将带你用C实现一个完整的TCP文件传输程序并通过Wireshark抓包工具逐帧解析从连接建立到文件落地的全过程。不同于单纯讲解API用法的教程我们将通过实验数据验证每个理论环节让你真正理解网络编程的底层逻辑。1. 实验环境搭建与工具准备在开始编码前我们需要配置好开发环境和抓包工具。这里选择Visual Studio 2019作为IDEWireshark 3.6作为网络分析工具。关键组件安装步骤如下Windows SDK安装确保勾选Windows 10 SDK和Debugging Tools for Windows需要WinSock2.h头文件和ws2_32.lib库支持Wireshark配置# 安装时需勾选Install WinPcap选项 # 抓包过滤器设置为tcp port 8401 ip.addr 127.0.0.1测试网络连通性// 快速测试Socket初始化 #include WinSock2.h #pragma comment(lib, ws2_32.lib) int main() { WSADATA wsaData; return WSAStartup(MAKEWORD(2,2), wsaData); }提示实验采用本地回环地址(127.0.0.1)进行通信避免防火墙干扰。实际开发中如需跨主机传输需确保双方网络策略允许指定端口通行。2. TCP三次握手的可视化验证让我们从最经典的三次握手开始。在服务器代码中关键调用顺序如下// 服务器端 SOCKET sock socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); bind(sock, (sockaddr*)addr, sizeof(addr)); listen(sock, 10); // 开始监听 SOCKET client accept(sock, NULL, NULL); // 阻塞等待连接对应的客户端连接代码// 客户端 SOCKET sock socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); connect(sock, (sockaddr*)addr, sizeof(addr)); // 触发握手在Wireshark中观察到的典型握手过程包序号方向标志位说明1客户端→服务端SYN1发送初始序列号J2服务端→客户端SYN1,ACK1确认J1发送初始序列号K3客户端→服务端ACK1确认K1连接建立完成关键发现通过对比代码执行时间戳和抓包时间戳可以验证accept()返回时刻正是第三次ACK到达之时。这解释了为什么服务器在accept()之前无法知道客户端信息。3. 文件传输的数据包解析文件传输的核心在于正确处理TCP的流式特性。我们采用分块传输策略// 发送方 while((bytesRead fread(buffer, 1, BUFFER_SIZE, file)) 0) { send(sock, buffer, bytesRead, 0); totalSent bytesRead; } // 接收方 while((bytesRecv recv(sock, buffer, BUFFER_SIZE, 0)) 0) { fwrite(buffer, 1, bytesRecv, file); totalReceived bytesRecv; }Wireshark抓包显示的重要特征MSS协商握手阶段通过TCP选项字段协商最大分段大小通常1460字节滑动窗口每个ACK包都包含窗口大小信息控制流量重传机制故意断开网络会观察到重传包其序列号与原始包相同注意实际传输中单个send()调用可能对应多个TCP包而单个recv()可能合并多个TCP包的数据。这是TCP的粘包特性导致的。4. 四次挥手的过程追踪连接关闭时的代码调用// 主动关闭方 shutdown(sock, SD_SEND); // 发送FIN while(recv(sock, buffer, BUFFER_SIZE, 0) 0); // 等待对方FIN closesocket(sock); // 被动关闭方 while(recv(sock, buffer, BUFFER_SIZE, 0) 0); // 读到EOF closesocket(sock); // 发送FINWireshark捕获的四次挥手过程第一次FIN主动方发送FIN序列号为N第一次ACK被动方立即确认N1第二次FIN被动方处理完数据后发送FIN序列号为M第二次ACK主动方确认M1进入TIME_WAIT状态常见误区很多人认为closesocket()会立即发送FIN。实际上如果发送缓冲区还有数据系统会先尝试发送剩余数据再发送FIN。5. 性能优化与错误处理基于抓包分析我们总结出几个关键优化点缓冲区大小设置// 理想缓冲区大小应与MSS匹配 int bufsize 1460 * 10; // 10个MSS setsockopt(sock, SOL_SOCKET, SO_SNDBUF, (char*)bufsize, sizeof(bufsize));Nagle算法控制int enable 1; // 禁用Nagle算法 setsockopt(sock, IPPROTO_TCP, TCP_NODELAY, (char*)enable, sizeof(enable));错误检测增强int error 0; socklen_t len sizeof(error); getsockopt(sock, SOL_SOCKET, SO_ERROR, (char*)error, len); if(error) { // 处理具体错误 }传输效率对比测试1MB文件优化措施传输时间(ms)数据包数量默认参数1250720调整缓冲区980690禁用Nagle算法850710综合优化6207006. 高级话题TCP状态机验证通过Wireshark的Follow TCP Stream功能可以完整观察TCP状态变迁连接建立CLOSED → SYN_SENT → ESTABLISHED连接关闭ESTABLISHED → FIN_WAIT_1 → FIN_WAIT_2 → TIME_WAIT特别值得注意的是TIME_WAIT状态。抓包显示即使程序已退出该连接仍会在系统中保持2MSL通常4分钟# 查看TCP状态 netstat -ano | findstr 8401 TCP 127.0.0.1:8401 127.0.0.1:62839 TIME_WAIT 07. 跨平台注意事项虽然本文以Windows为例但Linux/Mac下的开发有以下差异头文件差异// Linux #include sys/socket.h #include netinet/in.h #include unistd.h关闭连接// Linux使用close()而非closesocket() close(sockfd);错误处理// Linux使用errno而非WSAGetLastError() if (bytes -1) { perror(recv error); }在Linux下可以用tcpdump替代Wiresharktcpdump -i lo -nn tcp port 8401 -w capture.pcap通过这次实验最让我惊讶的是实际抓包数据与理论描述的精确对应。例如在测试时故意丢弃最后一个ACK确实观察到了服务端重传FIN包的现象。这种理论与实践的结合让原本抽象的网络协议变得触手可及。

相关文章:

从“三次握手”到文件落地:用Wireshark抓包带你彻底搞懂C++ Socket文件传输全过程

从“三次握手”到文件落地:用Wireshark抓包带你彻底搞懂C Socket文件传输全过程 当你在浏览器下载文件时,是否好奇过数据是如何跨越网络准确无误地到达你的电脑?本文将带你用C实现一个完整的TCP文件传输程序,并通过Wireshark抓包工…...

Step3-VL-10B-Base与C语言基础教程:嵌入式开发入门

Step3-VL-10B-Base与C语言基础教程:嵌入式开发入门 1. 引言 想学嵌入式开发但不知道从哪开始?很多新手卡在第一步:既要学C语言,又要懂硬件,感觉门槛很高。其实没那么复杂,用对方法就能快速上手。 这个教…...

【无线通信】基于统计信道的低复杂度旋转和位置优化为6D可移动天线无线通信附Matlab代码

✅作者简介:热爱科研的Matlab仿真开发者,擅长毕业设计辅导、数学建模、数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。🍎 往期回顾关注个人主页:Matlab科研工作室👇 关注我领取海量matlab电子书和…...

STM32CubeMX定时器避坑指南:为什么你的中断总是不触发?

STM32CubeMX定时器避坑指南:为什么你的中断总是不触发? 第一次使用STM32CubeMX配置定时器中断时,很多开发者都会遇到一个令人抓狂的问题——代码编译下载后,中断就像睡着了一样毫无反应。LED灯不闪烁、串口没输出、变量不更新&…...

Ubuntu常用的命令

ls -l # 输出当前文件夹下的所有文件的权限大小信息 ls -l 文件名 # 输出当前文件的权限大小信息 du -sh # 查看文件夹下所有文件的大小总和 df -h # 查看当前文件系统各分区的大小 hdparm -Tt /dev/sda1 # 查看分区磁盘的速度 ls -l | grep "^-" | wc -l # 当前目…...

PySR社区贡献指南:如何参与这个革命性符号回归开源项目的开发

PySR社区贡献指南:如何参与这个革命性符号回归开源项目的开发 【免费下载链接】PySR High-Performance Symbolic Regression in Python and Julia 项目地址: https://gitcode.com/gh_mirrors/py/PySR 想要为高性能符号回归工具PySR做出贡献吗?这份…...

StructBERT中文Large模型技术白皮书精读:结构化预训练策略深度解读

StructBERT中文Large模型技术白皮书精读:结构化预训练策略深度解读 1. 项目概述与核心价值 StructBERT是由阿里达摩院开发的中文预训练语言模型,它在经典BERT架构基础上引入了结构化预训练策略,显著提升了中文语言理解能力。这个模型特别针…...

OpenClaw安全防护指南:Qwen3-32B镜像对接时的权限控制策略

OpenClaw安全防护指南:Qwen3-32B镜像对接时的权限控制策略 1. 为什么需要安全防护? 去年我在尝试用OpenClaw自动整理财务报表时,曾因为一个简单的"读取桌面所有Excel文件"指令,差点导致包含客户隐私的文档被误传到测试…...

mrm-can-bus:轻量级嵌入式CAN设备服务协议库

1. 项目概述mrm-can-bus是一个面向嵌入式设备控制场景的轻量级 CAN 总线通信库,定位为“CAN Bus connectivity and local functions exposed via CAN Bus, common part”——即提供标准化的 CAN 连接能力,并将本地设备功能(如 GPIO 控制、ADC…...

Easy-Scraper:用 Rust 重新定义网页数据采集的效率边界

Easy-Scraper:用 Rust 重新定义网页数据采集的效率边界 【免费下载链接】easy-scraper Easy scraping library 项目地址: https://gitcode.com/gh_mirrors/ea/easy-scraper 当你需要从网页中提取数据时,是否遇到过这些困境:写了 200 行…...

从《巴伦周刊》谈起,我们该如何保住 SRE 的直觉?

大多数 AI 依然停留在执行层面,它们只能在 Demo 里写写脚本。一旦丢进真实的生产集群,面对复杂的资源依赖和权限限制,它们很难像人类专家那样,给出真正能拍板的建议。最近,《巴伦周刊》对 Chaterm 的报道引起了我的注意…...

Ubuntu服务器中文乱码终极解决方案:从locale配置到阿里云重启避坑指南

Ubuntu服务器中文乱码终极解决方案:从locale配置到阿里云重启避坑指南 当你第一次在Ubuntu服务器上看到中文字符变成一堆问号或方框时,那种困惑和挫败感我深有体会。特别是在云服务器环境下,问题往往比本地环境更复杂——即使按照常规教程操作…...

Gemma-3-12B-IT WebUI部署:支持HTTPS反向代理与Nginx负载均衡配置

Gemma-3-12B-IT WebUI部署:支持HTTPS反向代理与Nginx负载均衡配置 1. 项目概述 如果你正在寻找一个性能强大、易于部署,并且能通过Web界面直接对话的开源大模型,那么Gemma-3-12B-IT绝对值得关注。这个由Google开发的120亿参数模型&#xff…...

人工智能|大模型 —— 量化 —— 一文搞懂大模型量化技术:GGUF、GPTQ、AWQ

目前关于大模型量化技术的文章层出不穷,但对其理论部分的深入探讨却相对较少。本文将对大模型量化技术进行系统性的介绍,并重点聚焦于理论层面的深入解析。 一、大模型量化基础 大模型量化的核心在于将模型参数的精度从较高的位宽(bit-width…...

避坑指南:Windows下OpenCV摄像头索引混乱问题的3种解决之道

避坑指南:Windows下OpenCV摄像头索引混乱问题的3种解决之道 在工业视觉和智能监控领域,多摄像头协同工作是常见需求。但当你在Windows平台上使用OpenCV的VideoCapture接口时,可能会遇到这样的困扰:每次重启系统后,原本…...

告别低效循环:利用快马平台智能生成向量化代码,提升数据处理性能

最近在做一个数据分析项目时,遇到了性能瓶颈。处理一个几十万行的数据集时,简单的循环操作竟然要跑好几分钟。经过一番摸索,我发现向量化操作真是个神器,今天就分享一下如何用NumPy和Pandas来提升数据处理效率。 首先我们创建一个…...

nanobot实操手册:Qwen3-4B模型温度(temperature)、top_p、max_tokens参数详解

nanobot实操手册:Qwen3-4B模型温度(temperature)、top_p、max_tokens参数详解 1. nanobot简介与快速上手 nanobot是一款超轻量级的个人人工智能助手,灵感来源于OpenClaw项目。它最大的特点是代码量极小,仅需约4000行…...

【部署】windows下虚拟机OpenClaw Ubuntu 24.04.4 安装指南

未来已来,只需一句指令,养龙虾专栏导航,持续更新ing… 概述 前置环境:win10/11、vmware等虚拟机(安装时注意勾选VMware Tools、cpu可以分配2C,内存建议4G,硬盘空间建议给40G) 系统要求 Node.js 22+:安装脚本可自动检测并安装(下文补充手动安装方案); Ubuntu 24.0…...

OpenClaw本地模型成本对比:nanobot镜像vs商业API实测

OpenClaw本地模型成本对比:nanobot镜像vs商业API实测 1. 为什么需要关注OpenClaw的模型成本 上周我在尝试用OpenClaw自动化处理200份PDF文档时,意外发现账单上出现了三位数的API费用。这个数字让我意识到——当OpenClaw需要频繁调用大模型进行决策时&a…...

嵌入式C++ RAII互斥锁封装器MutexLocker详解

1. MutexLocker:嵌入式C RAII风格互斥锁封装器深度解析1.1 设计动机与工程价值在基于mbed RTOS(现为Mbed OS中CMSIS-RTOS v2兼容层)的嵌入式实时系统开发中,互斥量(Mutex)是保障多任务共享资源安全访问的核…...

NaViL-9B部署实操手册:supervisor服务管理+日志排查全流程详解

NaViL-9B部署实操手册:supervisor服务管理日志排查全流程详解 1. 平台简介 NaViL-9B是原生多模态大语言模型,支持纯文本问答和图片理解功能。该模型采用双24GB显卡配置,已预处理好模型权重和注意力机制兼容性问题,开箱即用。 2.…...

Java 设计模式・策略模式篇:从思想到代码实现

一、行为型模式 在面向对象的世界里,如何优雅地组织对象间的交互、分配职责,是每一位开发者都会反复思考的问题。直接硬编码交互逻辑固然简单,但当业务复杂度上升、对象协作关系变得错综复杂时,这种方式就会让代码变得僵化、难以…...

ECG-Emotion Recognition(情绪识别)实战指南:WESAD与DREAMER数据集深度解析与应用

1. 情绪识别与ECG技术入门指南 第一次接触ECG情绪识别时,我和大多数人一样充满疑惑:心跳数据真能反映人的情绪?经过三个月的项目实践,我可以肯定地说,ECG信号就像情绪的"心电图",愤怒时心跳加速、…...

【agent原理】OpenClaw之agent全链路详解

未来已来,只需一句指令,养龙虾专栏导航,持续更新ing… openclaw的术语约定 专业术语 类比 核心作用 不用的后果 Agent Bootstrapping AI员工的入职仪式 给AI办工牌、定岗位职责、录用户信息、建工作文件夹,只执行一次 手动建文件格式错乱、agent读不到规则、配置不统一、重…...

ImageSearch本地图片搜索引擎:从技术原理到实战应用

ImageSearch本地图片搜索引擎:从技术原理到实战应用 【免费下载链接】ImageSearch 基于.NET8的本地硬盘千万级图库以图搜图案例Demo和图片exif信息移除小工具分享 项目地址: https://gitcode.com/gh_mirrors/im/ImageSearch 价值定位:重新定义本地…...

vLLM-v0.17.1效果展示:vLLM支持MoE模型(如Mixtral)推理实测

vLLM-v0.17.1效果展示:vLLM支持MoE模型(如Mixtral)推理实测 1. vLLM框架核心能力 vLLM是一个专注于大语言模型推理的高性能服务库,最新发布的v0.17.1版本带来了对MoE(混合专家)架构模型的全面支持。这个最…...

突破透明动画性能瓶颈:VAP引擎实现移动端高效视觉体验

突破透明动画性能瓶颈:VAP引擎实现移动端高效视觉体验 【免费下载链接】vap VAP是企鹅电竞开发,用于播放特效动画的实现方案。具有高压缩率、硬件解码等优点。同时支持 iOS,Android,Web 平台。 项目地址: https://gitcode.com/gh_mirrors/va/vap …...

Webflux fromXXX对比

Mono.fromFuture和Mono.fromSupplier 刚开始尝试使用 Spring WebFlux 的时候,很多人都会使用 Mono.fromFuture() 将异步请求转成 Mono 对象,或者 Mono.fromSupplier() 将请求转成 MOno 对象,这两种方式在响应式编程 中都是不建议的&#xff0…...

FreeMoCap终极指南:如何用普通摄像头实现专业级3D动作捕捉

FreeMoCap终极指南:如何用普通摄像头实现专业级3D动作捕捉 【免费下载链接】freemocap Free Motion Capture for Everyone 💀✨ 项目地址: https://gitcode.com/GitHub_Trending/fr/freemocap 还在为专业动作捕捉设备的高昂价格而烦恼吗&#xff…...

家庭实验室:树莓派控制OpenClaw调用远程Qwen3-32B

家庭实验室:树莓派控制OpenClaw调用远程Qwen3-32B 1. 为什么选择树莓派OpenClaw组合 去年冬天,我在整理家庭实验室设备时发现一个闲置的树莓派4B。这台信用卡大小的电脑曾经用来跑Home Assistant控制智能家居,但后来换了NUC主机就被束之高阁…...