TCP三次握手与四次挥手——连接管理的“仪式感“

news2026/5/23 21:23:20
**导读**如果说HTTP是互联网世界的通用语言那么TCP就是支撑这一切的地下管道。但这条管道不是想通就通的——它有一套严格的礼仪规范也就是我们常说的三次握手和四次挥手。今天我们就来聊聊这场网络世界的相亲与分手大戏。标签TCP协议三次握手四次挥手网络抓包连接管理一、开场白为什么TCP要搞这么多仪式想象一下你给一个陌生人打电话上来就说喂我要跟你说个事对方可能会一脸懵逼“你谁啊信号好不好我要不要拿笔记一下”TCP面临的也是类似的问题。作为面向连接的可靠传输协议它必须解决三个核心问题双方是否都能收发数据——确认通信能力从哪个序号开始传——初始化序列号窗口大小是多少——协商接收能力这三个问题不解决后面的数据传输就是盲传丢包、乱序、重复包等问题会让你怀疑人生。所以TCP设计了一套握手机制在正式传数据之前先把这些家务事掰扯清楚。二、三次握手网络世界的相亲流程 生活化比喻TCP三次握手就像相亲——男方说你好我想认识你女方说好的我也想认识你男方说那我们开始吧。少了任何一步这亲都相不成。2.1 握手流程详解┌─────────────┐ ┌─────────────┐ │ 客户端 │ │ 服务端 │ │ (Client) │ │ (Server) │ └──────┬──────┘ └──────┬──────┘ │ │ │ ① SYN 1, seq x (ISN) │ │ ─────────────SYN────────────── │ │ 我想认识你我的初始序号是x │ │ │ │ ② SYN 1, ACK 1 │ │ seq y (ISN) │ │ ack x 1 │ │ ──────────SYNACK───────────── │ │ 我也想认识你我的初始序号是y │ │ 期待收到你的x1 │ │ │ │ ③ ACK 1, seq x 1 │ │ ack y 1 │ │ ─────────────ACK────────────── │ │ 好的我收到你的y了期待y1 │ │ │ │◄──────────连接建立开始数据传输──────────►│ │ │ 图1TCP三次握手流程图2.2 每一步都在干什么① 第一次握手SYN客户端发起请求客户端发送一个SYN包携带以下关键信息SYN 1表示这是一个同步请求seq x客户端的初始序列号ISN此时客户端进入SYN_SENT状态。这个ISNInitial Sequence Number初始序列号是怎么来的RFC 793规定ISN应该是一个随时间变化的计数器每4微秒加1。现代操作系统通常采用更复杂的随机算法以防止序列号预测攻击。② 第二次握手SYN-ACK服务端回应服务端收到SYN后如果同意建立连接会回复SYN-ACK包SYN 1我也要同步ACK 1确认收到你的SYNseq y服务端的初始序列号ack x 1期待收到客户端的下一个序号此时服务端进入SYN_RCVD状态。注意这里的ack x 1它表示我已经收到了序号x及之前的所有数据下次请从x1开始发。③ 第三次握手ACK客户端确认客户端收到SYN-ACK后发送最终的ACK包ACK 1确认收到seq x 1从x1开始发送数据ack y 1期待收到服务端的y1此时客户端进入ESTABLISHED状态。服务端收到ACK后也进入ESTABLISHED状态。2.3 为什么是三次两次不行吗这是一个经典的面试题。答案是两次握手无法防止历史重复连接初始化造成的混乱。假设只有两次握手客户端发送SYNseq100但由于网络延迟这个包滞留了客户端重发SYNseq200这次成功建立连接传输数据后断开此时滞留的第一个SYNseq100突然到达服务端服务端以为是新的连接请求回复SYN-ACK进入ESTABLISHED状态但客户端早已放弃这个连接不会回复导致服务端空等**⚠️ 关键区别**三次握手时服务端收到旧的SYN后回复SYN-ACK但客户端发现ack号不对期待的ack应该是201但收到的是101会发送RST包重置连接服务端因此不会进入ESTABLISHED状态。另外三次握手还能确保双方都有收发能力第一次服务端知道客户端能发第二次客户端知道服务端能收也能发第三次服务端知道客户端能收三、四次挥手优雅的分手仪式 生活化比喻四次挥手就像分手——一方说我们分手吧另一方说好的我知道了然后说那我也同意分手最后说再见。为什么分手比相亲多一步因为分手时可能还有话没说完。3.1 挥手流程详解┌─────────────┐ ┌─────────────┐ │ 客户端 │ │ 服务端 │ │ (Client) │ │ (Server) │ └──────┬──────┘ └──────┬──────┘ │ │ │ ① FIN 1, seq u │ │ ─────────────FIN────────────── │ │ 我要关闭发送通道了 │ │ │ │ ② ACK 1, seq v │ │ ack u 1 │ │ ────────────ACK────────────── │ │ 知道了等我发完数据 │ │ │ │ 【服务端继续发送未传完的数据】 │ │ │ │ ③ FIN 1, ACK 1 │ │ seq w │ │ ack u 1 │ │ ────────────FIN────────────── │ │ 我也发完了关闭吧 │ │ │ │ ④ ACK 1, seq u 1 │ │ ack w 1 │ │ ─────────────ACK────────────── │ │ 好的再见 │ │ │ │◄──────────连接关闭进入TIME_WAIT────────►│ │ │ 图2TCP四次挥手流程图3.2 为什么挥手要四次因为TCP连接是全双工的——数据可以同时在两个方向上传输。关闭连接时每个方向都需要单独关闭。步骤方向含义第一次FIN客户端 → 服务端客户端没有数据要发了第一次ACK服务端 → 客户端服务端知道客户端要关了第二次FIN服务端 → 客户端服务端也没有数据要发了第二次ACK客户端 → 服务端客户端知道服务端也要关了注意第二步和第三步之间服务端可能还有数据要发给客户端所以不能合并。只有当服务端也发完数据后才会发送自己的FIN。四、TIME_WAIT分手后的冷静期4.1 什么是TIME_WAIT主动关闭连接的一方通常是客户端在发送最后一个ACK后不会立即关闭而是进入TIME_WAIT状态等待2MSLMaximum Segment Lifetime最大报文生存时间后才真正关闭。MSL是什么MSL是IP数据包在网络中的最大生存时间RFC 793建议值为2分钟。Linux系统中MSL通常设置为30秒或1分钟因此2MSL就是1-2分钟。4.2 为什么要等2MSL有两个原因原因一确保最后一个ACK能被对方收到如果最后一个ACK丢失服务端会重发FIN包。客户端在TIME_WAIT状态下仍能收到这个FIN并回复ACK。如果没有TIME_WAIT客户端直接关闭服务端就会一直重发FIN陷入死循环。原因二防止旧的连接数据包干扰新连接假设没有TIME_WAIT客户端立即关闭并用相同端口建立新连接。此时网络中可能还有旧连接的延迟数据包到达会被新连接误收造成数据混乱。场景没有TIME_WAIT会发生什么 时间线 ─────────────────────────────────────────────────────────► T0: 客户端关闭连接端口8080释放 │ │ 网络中还有一个延迟的数据包在游荡 ▼ T1: 客户端立即用8080建立新连接 │ │ 延迟的数据包突然到达 ▼ T2: 新连接收到旧数据包 → 数据混乱 有了TIME_WAIT2MSL等待 ─────────────────────────────────────────────────────────► T0: 客户端进入TIME_WAIT端口8080仍被占用 │ │ 等待2MSL时间... │ 延迟的数据包要么到达被丢弃要么超时消失 ▼ T1: TIME_WAIT结束端口8080释放 │ ▼ T2: 新连接建立安全无干扰 图3TIME_WAIT的作用示意图4.3 TIME_WAIT过多的问题在高并发场景下如果短连接频繁创建和关闭可能导致大量端口处于TIME_WAIT状态耗尽可用端口。# 查看当前TIME_WAIT连接数 $ netstat -an | grep TIME_WAIT | wc -l 2847 # 查看各状态连接统计 $ netstat -n | awk /^tcp/ {S[$NF]} END {for(a in S) print a, S[a]} ESTABLISHED 156 TIME_WAIT 2847 CLOSE_WAIT 12解决方案连接池复用连接减少频繁创建销毁长连接HTTP Keep-Alive、TCP keepalive端口复用SO_REUSEADDR和SO_REUSEPORT调低TIME_WAIT时间谨慎操作修改tcp_tw_reuse和tcp_tw_recycle五、Wireshark实战亲眼见证握手与挥手5.1 环境准备我们需要一个可以观察TCP连接的场景。最简单的方法是访问一个HTTP网站或者自己写一个客户端/服务端程序。# 简单的Python HTTP服务器用于测试 # 服务端server.py import socket def start_server(host0.0.0.0, port8080): s socket.socket(socket.AF_INET, socket.SOCK_STREAM) s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) s.bind((host, port)) s.listen(5) print(fServer listening on {host}:{port}) while True: conn, addr s.accept() print(fConnection from {addr}) data conn.recv(1024) print(fReceived: {data.decode()}) conn.send(bHello, Client!) conn.close() print(Connection closed) if __name__ __main__: start_server()# 客户端client.py import socket import time def start_client(host127.0.0.1, port8080): s socket.socket(socket.AF_INET, socket.SOCK_STREAM) print(Creating socket...) # 连接服务器三次握手发生在这里 s.connect((host, port)) print(Connected to server!) # 发送数据 s.send(bHello, Server!) data s.recv(1024) print(fReceived: {data.decode()}) time.sleep(2) # 等待一会儿再关闭 # 关闭连接四次挥手发生在这里 s.close() print(Connection closed) if __name__ __main__: start_client()5.2 Wireshark抓包步骤Step 1启动Wireshark选择要监听的网络接口通常是Wi-Fi或以太网。Step 2设置过滤条件在过滤栏输入tcp.port 8080只显示8080端口的TCP流量。Step 3运行程序先启动服务端再启动客户端观察Wireshark捕获的包。5.3 分析抓包结果你应该能看到类似下面的包序列No. Time Source Destination Protocol Info 1 0.000 127.0.0.1 127.0.0.1 TCP 50000 → 8080 [SYN] Seq0 Win65535 2 0.001 127.0.0.1 127.0.0.1 TCP 8080 → 50000 [SYN, ACK] Seq0 Ack1 3 0.001 127.0.0.1 127.0.0.1 TCP 50000 → 8080 [ACK] Seq1 Ack1 4 0.002 127.0.0.1 127.0.0.1 TCP 50000 → 8080 [PSH, ACK] Seq1 Ack1 Len14 5 0.003 127.0.0.1 127.0.0.1 TCP 8080 → 50000 [PSH, ACK] Seq1 Ack15 Len14 6 2.005 127.0.0.1 127.0.0.1 TCP 50000 → 8080 [FIN, ACK] Seq15 Ack15 7 2.005 127.0.0.1 127.0.0.1 TCP 8080 → 50000 [ACK] Seq15 Ack16 8 2.006 127.0.0.1 127.0.0.1 TCP 8080 → 50000 [FIN, ACK] Seq15 Ack16 9 2.006 127.0.0.1 127.0.0.1 TCP 50000 → 8080 [ACK] Seq16 Ack16解析包1-3三次握手SYN → SYN-ACK → ACK包4-5数据传输PSH表示推送数据包6-9四次挥手FIN → ACK → FIN → ACK5.4 常用Wireshark过滤表达式# 只显示三次握手包 tcp.flags.syn1 tcp.flags.ack0 # SYN tcp.flags.syn1 tcp.flags.ack1 # SYN-ACK # 只显示四次挥手包 tcp.flags.fin1 # 只显示RST包异常断开 tcp.flags.reset1 # 显示特定端口的所有TCP包 tcp.port 8080 # 显示特定连接的包用stream index过滤 tcp.stream 0六、序列号与确认号TCP的记账本6.1 序列号Sequence NumberTCP把发送的每个字节都编上号这就是序列号。它不是给每个包编号而是给每个字节编号。**示例**假设ISN 1000发送Hello5个字节H → seq 1000e → seq 1001l → seq 1002l → seq 1003o → seq 1004确认号ACK应该是1005表示我已经收到1004及之前的所有字节期待1005。6.2 确认号Acknowledgment Number确认号表示期望收到的下一个字节的序号。如果收到ACK1005意味着发送方知道1004及之前的所有数据都已安全到达。6.3 累积确认TCP采用累积确认机制。即使中间某些ACK丢失只要后面的ACK到达就说明前面的数据也收到了。发送方发送 seq1000, len100 发送 seq1100, len100 发送 seq1200, len100 接收方收到 1000-1099 → 可能不立即ACK延迟ACK 收到 1100-1199 → 可能不立即ACK 收到 1200-1299 → 回复 ACK1300 发送方收到 ACK1300 → 知道1000-1299都已收到 图4累积确认机制示意图七、总结与思考7.1 核心要点回顾概念关键点三次握手SYN → SYN-ACK → ACK初始化序列号确认双向通信能力四次挥手FIN → ACK → FIN → ACK全双工连接需要分别关闭两个方向TIME_WAIT等待2MSL确保ACK到达防止旧数据干扰新连接序列号给每个字节编号实现可靠传输和顺序保证确认号期望收到的下一个字节序号采用累积确认7.2 常见面试题为什么三次握手不是两次为什么四次挥手不是三次TIME_WAIT状态的作用是什么如果服务端收到大量SYN包但不回复会发生什么SYN Flood攻击什么是TCP半连接队列和全连接队列 源码获取本文所有示例代码已整理到GitHub仓库包含Python TCP客户端/服务端示例Wireshark抓包分析脚本TIME_WAIT状态监控工具GitHub地址https://github.com/yourname/tcp-handshake-demo 思考题如果你用tcpdump抓包发现三次握手只看到了两个包可能是什么原因在Linux系统中如何查看和修改tcp_fin_timeout参数这个参数和TIME_WAIT有什么关系为什么HTTP/2和HTTP/3都在努力减少TCP连接的数量这与我们今天讲的握手/挥手成本有什么关系尝试用nc命令手动模拟TCP三次握手的过程提示需要用到nc -vz和抓包工具配合。 系列文章预告网络协议系列文章持续更新中01 | HTTP/1.1 vs HTTP/2 vs HTTP/3进化之路02 | HTTPS原理详解从明文到密文的蜕变03 | TCP三次握手与四次挥手——连接管理的仪式感本文04 | TCP拥塞控制网络堵车的解决方案05 | UDP简单但不可靠为什么还有人用06 | WebSocket全双工通信的实现原理关注专栏第一时间获取更新通知如果本文对你有帮助欢迎点赞、收藏、转发有任何问题欢迎在评论区留言讨论。标签TCP协议三次握手四次挥手网络抓包连接管理

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…