TCP三次握手四次挥手

一、TCP的三次握手

TCP是面向连接的传输协议,因此在使用TCP前必须先建立连接,而建立连接是通过三次握手来实现的。

                

三次握手具体实现过程:

        1、一开始,客户端和服务端都处于CLOSE状态,服务端主动监听某个端口,处于LISTEN状态。

        2、第一次握手:客户端会初始化随机序号,并发送一个SYN请求连接报文给服务端,此时客户端处于SYN-SENT状态(SYN发送状态)。

        3、第二次握手:服务端接收到客户端发来的SYN请求连接报文后,服务端也会随机初始化序号填入TCP首部的【序号】字段中,同时会把客户端发来的随机序列号+1作为确认应答报文ACK,然后将ACK和请求连接SYN一起发送给客户端,服务端此时处于SYN-RCVD状态(SYN收到状态)

        4、第三次握手:客户端收到服务端发来的ACK和SYN,此时客户端已经可以确认服务端能够正常接收和发送数据,客户端再发送一个确认应答报文ACK给服务端,这个报文可以携带客户端到服务端的数据(因为客户端经过前2次握手已经确认服务端接收和发送都是正常的)。客户端发送完后便处于ESTABLISHED状态(已经连接状态)。服务端接收到客户端发来的ACK后,也便进入ESTABLISHED状态(已经连接状态)。

        5、一旦完成三次握手,客户端和服务端都处于ESTABLISHED状态,此时双方都已建立连接,客户端和服务端就可以相互发送数据了。

二、为什么是三次握手?不是两次或者四次

为什么不是两次握手:如果只有两次握手,此时客户端接收到服务端发来的ACK和SYN,客户端可以确认服务端能够正常的接收和发送数据,而此时服务端无法确认客户端是否能正常接收和发送数据。

当客户端发送的SYN报文在网络中阻塞了,会重复发送多次SYN报文,那么服务端在收到请求后就会建立多个重复的无效链接,造成不必要的资源浪费。

为什么不是四次握手:四次握手的过程就是服务端在接收到客户端发来的SYN请求连接报文后,先发送一个确认应答报文ACK,然后在向客户端发送SYN请求连接报文,客户端收到SYN后向服务端发送一个ACK,到此便是四次握手的过程。

四次握手也能完成客户端与服务端建立连接,但四次握手中第二次与第三次可以优化成一次握手,提高通信效率,所以也就成了三次握手。

三、TCP三次握手是否每一次握手都能携带数据

结论:前两次握手不能携带数据,第三次握手客户端可以携带数据发送给服务端

原因:

1)如果第一次握手携带数据的话,此时客户端是不确定服务器是否能正常接收和发送数据。如果有人想要恶意攻击服务器,在第一次握手过程中,在SYN报文中放入大量数据,然后向服务端疯狂重复发送SYN报文块,此时服务端会浪费大量时间和内存空间来接收这些SYN报文块,此时服务器很容易受到攻击发生崩溃。

2)第二次握手过程中,服务器还无法确定客户端的接收和发送能力是否正常,当第二次握手过程中携带数据而此时如果ACK和SYN报文在网络阻塞中丢失了,会造成服务端重复发送数据,导致浪费大量资源和时间。

3)第三次握手过程中,客户端已经确认服务端能够正常接收和发送数据,并且客户端此时处于ESTABLISHED状态(已经连接状态),所以可以携带数据发送给服务端。

四、TCP四次挥手

TCP的断开是通过四次挥手来让客户端与服务端断开连接,其中客户端与服务端都可以主动断开连接。 以下是四次挥手的过程:

以客户端主动向服务端断开连接为例说明四次挥手过程:

        1、第一次挥手:客户端向服务端发送FIN结束报文段,报文中会指定一个序列号,客户端便会进入FIN-WAIT1状态

        2、第二次挥手:服务端接收到客户端发来的FIN,会向客户端发送一个ACK应答报文,此时服务端便会进入CLOSE-WAIT状态

        3、客户端接收到服务端发来的ACK应答报文后,便进入FIN-WAIT2状态

        4、第三次挥手:服务端也向客户端发送一个FIN结束报文段,此时服务端进入LAST-ACK状态

        5、第四次挥手:客户端接收到服务端发来的FIN,会向服务端发送一个确认应答报文ACK,之后便进入TIME-WAIT状态

        6、当服务端接收到客户端发来的ACK之后,便进入CLOSE状态,自此服务端关闭连接状态

        7、 客户端经过一段时间后,自动进入CLOSE状态,自此客户端也关闭连接状态

五、TCP四次挥手能否合并三次挥手

在TCP三次握手的过程中,ACK和SYN是由操作系统内核来完成的,可以同一时间发送,

在TCP四次挥手的过程中,ACK是由操作系统内核完成,ACK在接收到FIN后会第一时间返回,而FIN是由应用程序代码完成,只有在调用socket的close()方法时才会触发FIN,因此在TCP四次挥手中ACK和FIN不是同一时间发送,因此TCP的四次挥手不能合并成三次挥手。
 

### TCP 三次握手四次挥手的过程及原理 #### 一、TCP 三次握手过程及原理 TCP 是一种面向连接的可靠传输协议,在数据传输前需要通过三次握手来建立连接。以下是其具体过程: 1. **第一次握手** 客户端向服务器发起连接请求,发送一个带有 SYN 标志位的数据包,并随机生成一个初始序列号 `Seq=x`,此时客户端进入 `SYN_SENT` 状态[^5]。 2. **第二次握手** 服务器收到客户端的 SYN 请求后,返回一个确认消息,该消息同时携带两个标志位:SYN 和 ACK。其中,SYN 表示同意建立连接并随机生成自己的初始序列号 `Seq=y`;ACK 则是对客户端序列号的确认,即 `Ack=x+1`。此时服务器进入 `SYN_RCVD` 状态[^4]。 3. **第三次握手** 客户端收到服务器的响应后,再次发送一个确认消息给服务器,此消息仅带有一个 ACK 标志位,用于确认服务器的序列号,即 `Ack=y+1`。随后,客户端进入 `ESTABLISHED` 状态,而服务器在接收到这个 ACK 后也进入 `ESTABLISHED` 状态,至此完成三次握手[^1]。 通过以上三个步骤,双方不仅建立了连接,还同步了各自的序列号和确认号,从而确保后续通信的可靠性[^3]。 --- #### 二、TCP 四次挥手过程及原理 由于 TCP 协议采用全双工模式工作,因此关闭连接时需分别处理双向通道的释放问题,这通常涉及四个阶段的操作,称为“四次挥手”。 1. **第一次挥手** 主动关闭的一方(通常是客户端)发送 FIN 报文通知对方自己已经没有数据要发送了,同时终止本方向上的数据流。此时,主动关闭方进入 `FIN_WAIT_1` 状态[^2]。 2. **第二次挥手** 被动关闭一方(通常是服务器)收到 FIN 报文后,回应一个 ACK 报文作为确认,设置 `Ack=seq+1`,并将被动关闭方的状态改为 `CLOSE_WAIT`。与此同时,主动关闭方则转变为 `FIN_WAIT_2` 状态[^4]。 3. **第三次挥手** 当被关闭方也没有更多数据待发时,它会再发送一个 FIN 报文告知主动关闭方准备完全断开连接。此时,被动关闭方变为 `LAST_ACK` 状态[^5]。 4. **第四次挥手** 主动关闭方接收到来自被动关闭方的 FIN 报文之后,回复最后一个 ACK 报文予以确认,然后进入 `TIME_WAIT` 状态。经过一段时间延迟(一般为两倍最大分组生存时间 MSL),如果未检测到异常情况,则正式关闭连接并转入 `CLOSED` 状态。同样地,被动关闭方在接受到最终 ACK 报文后立即切换至 `CLOSED` 状态,结束整个流程[^3]。 值得注意的是,“四次挥手”的核心在于允许两端独立管理各自方向上的资源回收,保障任何尚未送达的信息能够得到妥善处理[^2]。 --- ```python # Python模拟TCP三次握手四次挥手逻辑示意 class TCPSession: def __init__(self, client_seq, server_seq): self.client_seq = client_seq self.server_seq = server_seq def three_way_handshake(self): print(f"Client -> Server: SYN Seq={self.client_seq}") self.client_seq += 1 print(f"Server <- Client: SYN+ACK Ack={self.client_seq} Seq={self.server_seq}") self.server_seq += 1 print(f"Client -> Server: ACK Ack={self.server_seq}") def four_way_wavehand(self): print(f"Client -> Server: FIN Seq={self.client_seq}") print(f"Server <- Client: ACK Ack={self.client_seq + 1}") print(f"Server -> Client: FIN Seq={self.server_seq}") print(f"Client <- Server: ACK Ack={self.server_seq + 1}") session = TCPSession(0, 0) print("\n--- Three-Way Handshake ---") session.three_way_handshake() print("\n--- Four-Way Wavehand ---") session.four_way_wavehand() ``` ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值