- TCP 三次/四次握手
TCP 建立连接 - 3 次握手
建立过程:
- client 发送
SYN=1, seq=x
:SYN=1
建立连接的标识,seq=x
是客户端初始化的序号 - server 发送
SYN=1, seq=y, ACK=1, ack=x+1
:seq=y
服务端初始化的序号,ACK=1
确认字符,ack=x+1
确任序号(=clientSeq+1) - client 发送
ACK=1, ack=y+1, seq=x+1
:ACK=1
确认字符,ack=y+1
确任序号(=serverSeq+1)
Note:
Q:为什么要 3 次握手,而不是 2 次握手?
A:3 次握手能够防止 client 发送的失效连接被 server 响应,而重新建立连接,避免资源的浪费。通过 server 发送的 ack=clientSeq+1,client 可以校验 clientSeq,从而确认是否发送第三次握手。
TCP 释放连接 - 4 次握手
释放过程:
- client send
FIN=1, seq=u
:FIN=1
关闭连接标识,seq=u
client 初始化序列号 - server send
ACK=1, seq=v, ack=u+1
:ACK
确认字符,ack=u+1
确认序号=clientSeq+1,seq=v
server 端初始化序号 - client 端虽然不再发送数据了,但是可以接收数据。server 端此时仍可以向 client 传输数据
- server send
FIN=1, ACK=1, seq=w, ack=u+1
:FIN=1
关闭连接标识,ACK=1
确认字符,ack=u+1
确认序号=clientSeq+1,seq=w
server 发送序号 - client send
ACK=1, seq=u+1, ack=w=1
:ACK=1
确认字符,ack=w+1
确认序号=serverSeq+1,seq=u+1
client 端发送序号
Note:
-
Q:为什么关闭连接需要 4 次握手,而建立连接只需要 3 次?
A:建立连接时,标识(SYN) 和 ACK 是一起发送的,而关闭连接时,标识(FIN) 和 ACK 是分开发送的。分开发送的原因是,虽然 client 端表明其准备关闭连接了,只是表明 client 不再发送数据,但其仍可以接收数据。server 端仍可以在确认收到 client 端关闭连接的消息后,将未发送完的数据发送给 client 端,之后再发送关闭连接标识(FIN)