元素码农
基础
UML建模
数据结构
算法
设计模式
网络
TCP/IP协议
HTTPS安全机制
WebSocket实时通信
数据库
sqlite
postgresql
clickhouse
后端
rust
go
java
php
mysql
redis
mongodb
etcd
nats
zincsearch
前端
浏览器
javascript
typescript
vue3
react
游戏
unity
unreal
C++
C#
Lua
App
android
ios
flutter
react-native
安全
Web安全
测试
软件测试
自动化测试 - Playwright
人工智能
Python
langChain
langGraph
运维
linux
docker
工具
git
svn
🌞
🌙
目录
▶
网络层
▶
IP协议
IP数据报格式
子网划分原理
CIDR无类寻址
IP分片与重组
IP选项字段
▶
ICMP协议
差错报文类型
Ping实现原理
Traceroute原理
▶
ARP协议
地址解析原理
ARP缓存表
代理ARP
▶
路由协议
RIP协议详解
OSPF协议原理
BGP协议机制
▶
IPv6协议
IPv6地址结构
IPv6报文格式
IPv6扩展头
IPv6过渡技术
▶
移动IP
移动IP原理
代理发现机制
注册与路由优化
▶
网络QoS
QoS服务模型
IntServ架构
DiffServ架构
MPLS技术
▶
NAT技术
NAT原理与类型
NAT穿透技术
NAT64与DNS64
▶
组播路由
组播基础
IGMP协议
PIM协议族
▶
网络安全
IPSec协议族
VPN技术详解
DDoS防护
▶
传输层
▶
TCP协议
三次握手机制
滑动窗口原理
拥塞控制算法
四次挥手过程
超时重传机制
TCP状态转换
快速重传机制
快速恢复算法
选择性确认SACK
时间戳选项
▶
UDP协议
数据报结构解析
实时传输优化
UDP校验和计算
UDP广播与多播
UDP性能调优
UDP可靠传输
▶
SCTP协议
SCTP协议概述
多宿主支持
多流传输
关联建立过程
心跳机制
▶
传输层安全
TLS协议详解
DTLS协议
密钥交换机制
证书验证
▶
应用层
▶
HTTP协议
请求响应模型
持久连接机制
HTTP消息格式
状态码详解
Cookie机制
HTTP缓存机制
HTTP认证机制
HTTPS详解
TLS/SSL协议
HTTP/1.0详解
HTTP/1.1详解
HTTP/2详解
HTTP/3详解
▶
DNS系统
域名解析过程
记录类型详解
递归与迭代查询
DNS缓存机制
▶
FTP协议
FTP工作原理
主动与被动模式
FTP命令详解
▶
SMTP协议
邮件传输流程
SMTP会话过程
邮件格式规范
▶
DHCP协议
DHCP工作原理
地址分配过程
DHCP中继代理
▶
SNMP协议
SNMP架构
MIB数据库
SNMP操作
SNMPv3安全机制
▶
WebSocket协议
WebSocket原理
握手升级机制
数据帧格式
心跳与连接维护
▶
QUIC协议
QUIC协议特性
0-RTT建连
多路复用
丢包恢复
▶
gRPC协议
gRPC基础
服务定义
通信模式
负载均衡
发布时间:
2025-03-25 08:33
↑
☰
# TCP四次挥手过程 TCP连接的终止过程比建立连接更复杂,需要经过四次握手(也称为四次挥手)才能完全关闭连接。这种设计确保了双方都能够完整地传输和接收所有数据。 ## 为什么需要四次挥手 在TCP连接中,连接的每一端都有发送和接收两个通道。四次挥手的设计允许这两个通道可以独立关闭,这样即使一方停止发送数据,也可以继续接收对方发送的数据。这种机制保证了: 1. 数据传输的完整性 2. 避免数据丢失 3. 双方都能优雅地关闭连接 ## 四次挥手的详细过程 ```mermaid sequenceDiagram participant C as 客户端 participant S as 服务器 Note over C,S: 连接已建立(ESTABLISHED) C->>S: FIN=1, seq=x Note over C: FIN_WAIT_1 S->>C: ACK=1, ack=x+1 Note over C: FIN_WAIT_2 Note over S: CLOSE_WAIT S->>C: FIN=1, seq=y Note over S: LAST_ACK C->>S: ACK=1, ack=y+1 Note over C: TIME_WAIT Note over C: 等待2MSL Note over C,S: 连接关闭(CLOSED) ``` ### 第一次挥手(FIN_WAIT_1) 1. 主动关闭方(通常是客户端)发送FIN报文 - FIN=1,表示要关闭发送通道 - 序列号seq=x - 进入FIN_WAIT_1状态 ### 第二次挥手(FIN_WAIT_2) 1. 被动关闭方(通常是服务器)收到FIN后 - 发送ACK确认 - 确认号ack=x+1 - 主动方收到ACK后进入FIN_WAIT_2状态 - 被动方进入CLOSE_WAIT状态 ### 第三次挥手(LAST_ACK) 1. 被动关闭方处理完所有数据后 - 发送FIN报文 - FIN=1,序列号seq=y - 进入LAST_ACK状态 ### 第四次挥手(TIME_WAIT) 1. 主动关闭方收到FIN后 - 发送最后的ACK确认 - 确认号ack=y+1 - 进入TIME_WAIT状态 - 等待2MSL时间后关闭连接 ## TIME_WAIT状态的重要性 ### 为什么需要TIME_WAIT 1. 确保最后的ACK能够到达 - 如果ACK丢失,对方会重发FIN - TIME_WAIT期间可以重发ACK 2. 防止延迟报文影响新连接 - 等待2MSL可以让旧连接的报文在网络中消失 - 避免新老连接的报文混淆 ### MSL(Maximum Segment Lifetime) - MSL是报文在网络中的最大生存时间 - 通常为30秒或1分钟 - TIME_WAIT持续时间为2MSL - 在Linux系统中可以通过tcp_max_tw_buckets参数调整 ## 异常情况处理 ### 1. 提前关闭CLOSE_WAIT 如果服务器在CLOSE_WAIT状态时崩溃: - 客户端FIN_WAIT_2状态超时 - 系统自动关闭连接 ### 2. RST报文 在以下情况会发送RST报文: - 访问不存在的端口 - 连接超时 - 应用程序异常终止 ### 3. 同时关闭 双方同时发送FIN报文时: - 都进入FIN_WAIT_1状态 - 经过交换ACK后进入CLOSING状态 - 最终都进入TIME_WAIT状态 ## 编程实践 ### 1. 优雅关闭连接 ```c // 发送剩余数据 send(sockfd, remain_data, len, 0); // 关闭发送通道 shutdown(sockfd, SHUT_WR); // 接收对方剩余数据 while (recv(sockfd, buffer, sizeof(buffer), 0) > 0) { // 处理数据 } // 完全关闭连接 close(sockfd); ``` ### 2. 处理TIME_WAIT ```c // 允许地址重用 int reuse = 1; setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)); ``` ## 常见问题与解决方案 ### 1. CLOSE_WAIT积累 问题:服务器出现大量CLOSE_WAIT状态 解决方案: - 检查应用程序是否正确关闭连接 - 实现连接超时机制 - 使用连接池管理连接 ### 2. TIME_WAIT过多 问题:服务器出现大量TIME_WAIT状态 解决方案: - 开启SO_REUSEADDR选项 - 使用长连接代替短连接 - 调整内核参数tcp_max_tw_buckets ### 3. 连接关闭不完全 问题:连接没有正确关闭 解决方案: - 使用shutdown()代替close() - 实现完整的错误处理 - 添加连接状态监控 ## 调试技巧 ### 1. 使用tcpdump抓包分析 ```bash # 抓取TCP连接关闭过程的报文 tcpdump -i any tcp port 80 and '(tcp[tcpflags] & (tcp-fin|tcp-syn|tcp-rst) != 0)' ``` ### 2. 使用netstat监控连接状态 ```bash # 查看各种状态的连接数量 netstat -n | awk '/^tcp/ {++S[$NF]} END {for(a in S) print a, S[a]}' ``` ## 总结 TCP的四次挥手过程是一个精心设计的连接终止机制,它保证了: 1. 数据传输的可靠性和完整性 2. 连接的优雅关闭 3. 避免旧连接影响新连接 在实际应用中,应该: 1. 正确实现连接关闭逻辑 2. 合理处理异常情况 3. 定期监控连接状态 4. 根据具体场景优化连接管理策略