说一下 TCP 的三次握手四次挥手过程

简介: 我是小假 期待与你的下一次相遇 ~

第一部分:核心概念与背景

在开始之前,先明确几个关键点:

  1. TCP 是什么:传输控制协议,是一种面向连接的、可靠的、基于字节流的传输层通信协议。
  2. “连接”是什么:TCP 的连接并非物理上的电路,而是通信双方在内存中维护的一种状态信息(如序列号、窗口大小等)。建立连接就是同步双方的初始状态
  3. 标志位:TCP 头部中有几个重要的控制位,在握手挥手中起到关键作用:
  • SYN:同步序列号,用于建立连接
  • ACK:确认,表示确认号字段有效。
  • FIN:终止,用于关闭连接
  1. 序列号与确认号
  • 序列号:标识发送的数据字节流的顺序。
  • 确认号:期望收到对方下一个报文段的第一个数据字节的序列号,表示此序号之前的数据已成功接收。

第二部分:三次握手 — 建立连接

三次握手的根本目的是:同步双方的初始序列号,并交换其他参数,为可靠数据传输做准备。同时,它也证明了双方都具有发送和接收的能力。

假设客户端(Client)主动发起连接,服务器(Server)等待连接。

第一步:第一次握手(Client -> Server)

  • 客户端发送一个 TCP 报文段。
  • 设置标志位 SYN = 1,表示这是连接请求。
  • 同时,客户端会随机生成一个初始序列号 client_isn,放在序列号字段中。
  • 此时不携带应用层数据。
  • 客户端状态:从 CLOSED 进入 SYN-SENT(同步已发送)。

第二步:第二次握手(Server -> Client)

  • 服务器收到 SYN 报文后,如果同意建立连接,则会回复一个报文段。
  • 设置标志位 SYN = 1 ACK = 1
  • 服务器也会随机生成自己的初始序列号 server_isn,放在序列号字段中。
  • 确认号字段设置为 client_isn + 1,表示“我已收到你的序列号为 client_isn 的 SYN 包,期待下一个数据从 client_isn + 1 开始”。
  • 此时可以携带或不携带应用层数据(但通常不携带)。
  • 服务器状态:从 LISTEN 进入 SYN-RCVD(同步已收到)。

第三步:第三次握手(Client -> Server)

  • 客户端收到服务器的 SYN-ACK 报文后,会再发送一个确认报文。
  • 设置标志位 ACK = 1
  • 序列号字段设置为 client_isn + 1(因为第一次握手消耗了一个序列号)。
  • 确认号字段设置为 server_isn + 1,表示“我已收到你的序列号为 server_isn 的 SYN 包”。
  • 此报文可以携带应用层数据(连接建立后即可开始传输)。
  • 客户端状态:从 SYN-SENT 进入 ESTABLISHED(已建立连接)。
  • 服务器收到这个 ACK 后,状态也从 SYN-RCVD 进入 ESTABLISHED

为什么是三次,而不是两次?

  • 根本原因防止已失效的连接请求报文突然到达,导致资源浪费和错误。
  • 经典场景:客户端发送了一个 SYN 报文,但由于网络拥堵,迟迟未到达服务器。客户端超时重传了一个新的 SYN 并成功建立了连接、传输数据、关闭连接。此时,那个失效的旧 SYN 报文终于到达了服务器。如果只用两次握手,服务器会认为这是一个新的连接请求,直接进入 ESTABLISHED 状态,并分配资源等待客户端发送数据,但客户端早已关闭,不会发送任何数据。这就造成了服务器的资源被白白占用
  • 三次握手的作用:在两次握手的情况下,服务器会直接建立连接。而采用三次握手,服务器在收到失效的 SYN 后,会回复 SYN-ACK,但客户端(因为已经关闭)不会回复最终的 ACK,因此服务器在 SYN-RCVD 状态等待一段时间后,会因收不到 ACK 而关闭这个半连接,从而避免资源浪费。

第三部分:四次挥手 — 关闭连接

四次挥手的目的是:双方都确认没有数据需要发送了,然后安全地关闭连接。由于 TCP 连接是全双工的(可以同时独立地进行双向数据传输),因此每个方向都必须单独关闭。

假设客户端主动发起关闭。

第一步:第一次挥手(Client -> Server)

  • 客户端应用程序调用 close() 方法,发送一个 TCP 报文段。
  • 设置标志位 FIN = 1
  • 携带一个当前的序列号 seq = u
  • 客户端状态:从 ESTABLISHED 进入 FIN-WAIT-1(终止等待1)。

第二步:第二次挥手(Server -> Client)

  • 服务器收到 FIN 报文后,立即回复一个确认报文。
  • 设置标志位 ACK = 1
  • 确认号设置为 u + 1
  • 此时,从客户端到服务器方向的连接就关闭了。客户端不会再向服务器发送数据,但服务器可能还有数据要发送给客户端,连接处于半关闭状态
  • 服务器状态:从 ESTABLISHED 进入 CLOSE-WAIT(关闭等待)。
  • 客户端状态:收到这个 ACK 后,从 FIN-WAIT-1 进入 FIN-WAIT-2(终止等待2)。

第三步:第三次挥手(Server -> Client)

  • 当服务器将剩余数据全部发送完毕后,它的应用程序也会调用 close() 方法,发送一个 FIN 报文。
  • 设置标志位 FIN = 1(通常还会同时设置 ACK = 1)。
  • 携带一个序列号 seq = w
  • 服务器状态:从 CLOSE-WAIT 进入 LAST-ACK(最后确认)。

第四步:第四次挥手(Client -> Server)

  • 客户端收到服务器的 FIN 报文后,必须发送一个确认报文。
  • 设置标志位 ACK = 1
  • 确认号设置为 w + 1
  • 客户端状态:从 FIN-WAIT-2 进入 TIME-WAIT(时间等待)。此时客户端连接尚未完全关闭,需要等待 2MSL(两倍的最大报文段生存时间)时长。
  • 服务器收到这个 ACK 后,状态从 LAST-ACK 进入 CLOSED,连接正式关闭。

为什么客户端需要 TIME-WAIT 状态?等待 2MSL 的目的是什么?

  1. 可靠地终止连接:确保客户端发送的最后一个 ACK 能到达服务器。如果这个 ACK 丢失,服务器会超时重传 FIN 报文。客户端在 TIME-WAIT 状态下可以收到这个重传的 FIN,并重发 ACK。
  2. 让旧的重复报文在网络中消逝:等待 2MSL 时间,足以让本次连接过程中产生的所有报文都在网络中消失。这样,就不会影响后续使用相同四元组(源IP、源端口、目的IP、目的端口)的新连接。

总结与对比

阶段

目的

关键标志位

状态变化

三次握手

建立连接,同步初始序列号

SYN, ACK

CLOSED -> SYN-SENT -> SYN-RCVD -> ESTABLISHED

第一次挥手

客户端发起关闭(关闭己方发送通道)

FIN

ESTABLISHED -> FIN-WAIT-1

第二次挥手

服务器确认收到关闭请求

ACK

ESTABLISHED -> CLOSE-WAIT / FIN-WAIT-1 -> FIN-WAIT-2

第三次挥手

服务器发起关闭(关闭己方发送通道)

FIN, ACK

CLOSE-WAIT -> LAST-ACK

第四次挥手

客户端确认收到关闭请求

ACK

FIN-WAIT-2 -> TIME-WAIT -> CLOSED

面试回答

三次握手(建立连接)

目的:为了确认双方的发送和接收能力都正常,并同步初始序列号(ISN)。

  1. 第一次握手(SYN):客户端发送一个 SYN的包,并随机生成一个初始序列号 X 给服务器。意思是:“我想建立连接,这是我的起始号。”
  2. 第二次握手(SYN+ACK):服务器收到后,如果同意连接,会回复一个 SYNACK的包。它会确认客户端的序列号,同时自己也随机生成一个初始序列号 Y。意思是:“我收到你的请求了,我同意连接。这是我的起始号。”
  3. 第三次握手(ACK):客户端收到回复后,再向服务器发送一个 ACK 的包,确认服务器的序列号。意思是:“收到你的同意了,连接建立成功。”
    至此,连接建立,可以开始数据传输。

四次挥手(断开连接)

目的:双方都确认数据已发送完毕,可以安全关闭连接。之所以是四次,因为TCP连接是全双工的,每一方向都需要独立关闭。

  1. 第一次挥手(FIN):假设客户端数据发完了,它会发送一个 FIN 的包,并携带一个序列号。意思是:“我这边数据发完了,要关闭连接了。”
  2. 第二次挥手(ACK):服务器收到 FIN 后,会立刻回复一个 ACK 的确认包。意思是:“我知道你要关了,但我可能还有数据没发完。”
    此时,客户端到服务器的连接关闭,但服务器到客户端的连接依然可以继续发送数据。
  3. 第三次挥手(FIN):当服务器也把所有数据发完后,它会发送自己的 FIN 包。意思是:“我这边数据也发完了,我也要关了。”
  4. 第四次挥手(ACK):客户端收到服务器的 FIN 后,会发送最后一个 ACK 的确认包。意思是:“好的,我们都关了。”
    客户端会等待一段时间(2MSL)后彻底关闭,服务器收到这个ACK后立即关闭。连接完全断开。

所以总结一下,握手是三次,因为第二次的回复合并了确认和发起请求。而挥手是四次,因为服务器在收到关闭请求后,可能还有数据要发送,所以把确认和发起自己的关闭请求分成了两步,没办法合并。

追加为什么握手是三次,不是两次?

主要是为了防止已失效的连接请求报文突然又传到服务器,导致服务器错误地打开连接。三次握手机制下,这个失效的报文会让客户端收到一个它并未请求的确认,客户端会拒绝,服务器收不到第三次ACK,就不会建立连接。

追加:为什么挥手需要四次?

因为TCP是全双工的。当一方说“我要关了”(发FIN),只代表它不再发送数据,但还可以接收数据。另一方可能还有数据要发送,所以必须先立刻回复一个ACK,等自己数据发完再发FIN。因此ACK和FIN在多数情况下不能像握手那样合并发送。

追加: TIME_WAIT状态是什么?为什么需要等待2MSL?

主动关闭的一方(先发FIN的,比如刚才例子里的客户端)在发送完最后一个ACK后,会进入TIME_WAIT状态,等待2倍的最大报文段生存时间(2MSL)
两个目的

  1. 确保最后的ACK能到达:如果这个ACK丢失了,对方会超时重发FIN,TIME_WAIT状态下的客户端还能重发ACK。
  2. 让本次连接产生的所有报文在网络中都消失,避免影响后续新建的相同端口的连接。
相关文章
|
9天前
|
人工智能 JavaScript Linux
【Claude Code 全攻略】终端AI编程助手从入门到进阶(2026最新版)
Claude Code是Anthropic推出的终端原生AI编程助手,支持40+语言、200k超长上下文,无需切换IDE即可实现代码生成、调试、项目导航与自动化任务。本文详解其安装配置、四大核心功能及进阶技巧,助你全面提升开发效率,搭配GitHub Copilot使用更佳。
|
3天前
|
JSON API 数据格式
OpenCode入门使用教程
本教程介绍如何通过安装OpenCode并配置Canopy Wave API来使用开源模型。首先全局安装OpenCode,然后设置API密钥并创建配置文件,最后在控制台中连接模型并开始交互。
1503 5
|
11天前
|
存储 人工智能 自然语言处理
OpenSpec技术规范+实例应用
OpenSpec 是面向 AI 智能体的轻量级规范驱动开发框架,通过“提案-审查-实施-归档”工作流,解决 AI 编程中的需求偏移与不可预测性问题。它以机器可读的规范为“单一真相源”,将模糊提示转化为可落地的工程实践,助力开发者高效构建稳定、可审计的生产级系统,实现从“凭感觉聊天”到“按规范开发”的跃迁。
1659 17
|
9天前
|
人工智能 JavaScript 前端开发
【2026最新最全】一篇文章带你学会Cursor编程工具
本文介绍了Cursor的下载安装、账号注册、汉化设置、核心模式(Agent、Plan、Debug、Ask)及高阶功能,如@引用、@Doc文档库、@Browser自动化和Rules规则配置,助力开发者高效使用AI编程工具。
1264 6
|
10天前
|
消息中间件 人工智能 Kubernetes
阿里云云原生应用平台岗位急招,加入我们,打造 AI 最强基础设施
云原生应用平台作为中国最大云计算公司的基石,现全面转向 AI,打造 AI 时代最强基础设施。寻找热爱技术、具备工程极致追求的架构师、极客与算法专家,共同重构计算、定义未来。杭州、北京、深圳、上海热招中,让我们一起在云端,重构 AI 的未来。
|
13天前
|
IDE 开发工具 C语言
【2026最新】VS2026下载安装使用保姆级教程(附安装包+图文步骤)
Visual Studio 2026是微软推出的最新Windows专属IDE,启动更快、内存占用更低,支持C++、Python等开发。推荐免费的Community版,安装简便,适合初学者与个人开发者使用。
1288 12
|
7天前
|
云安全 安全
免费+限量+领云小宝周边!「阿里云2026云上安全健康体检」火热进行中!
诚邀您进行年度自检,发现潜在风险,守护云上业务连续稳健运行
1177 2
|
13天前
|
人工智能 测试技术 开发者
AI Coding后端开发实战:解锁AI辅助编程新范式
本文系统阐述了AI时代开发者如何高效协作AI Coding工具,强调破除认知误区、构建个人上下文管理体系,并精准判断AI输出质量。通过实战流程与案例,助力开发者实现从编码到架构思维的跃迁,成为人机协同的“超级开发者”。
1008 94
|
9天前
|
人工智能 JSON 自然语言处理
【2026最新最全】一篇文章带你学会Qoder编辑器
Qoder是一款面向程序员的AI编程助手,集智能补全、对话式编程、项目级理解、任务模式与规则驱动于一体,支持模型分级选择与CLI命令行操作,可自动生成文档、优化提示词,提升开发效率。
784 8
【2026最新最全】一篇文章带你学会Qoder编辑器