先亮出这篇文章的思维导图
UDP
相比,
TCP
有三大核心特性:
面向连接。所谓的连接,指的是客户端和服务器的连接,在双方互相通信之前,TCP 需要三次握手建立连接,而 UDP 没有相应建立连接的过程。
可靠性。TCP 花了非常多的功夫保证连接的可靠,这个可靠性体现在哪些方面呢?一个是有状态,另一个是可控制。
无状态
,
不可控
的。
爱
的能力。
爱
和
被爱
的能力。
被爱
的能力。
爱
和
被爱
的能力,两人开始一段甜蜜的爱情。
发送的能力
和
接收的能力
。于是便会有下面的三次握手的过程:
CLOSED
状态。然后服务端开始监听某个端口,进入了
LISTEN
状态。
SYN-SENT
状态。
SYN
和
ACK
(对应客户端发来的SYN),自己变成了
SYN-REVD
。
ACK
给服务端,自己变成了
ESTABLISHED
状态;服务端收到
ACK
之后,也变成了
ESTABLISHED
状态。
凡是需要对端确认的,一定消耗TCP报文的序列号。
发送
和
接收
的能力,那四次握手可以嘛?
ESTABLISHED
状态,并且已经能够确认服务器的接收、发送能力正常,这个时候相对安全了,可以携带数据。
SYN
报文,状态变化会是怎样的呢?
SYN
报文的同时,接收方也给发送方发
SYN
报文,两个人刚上了!
SYN
,两者的状态都变为
SYN-SENT
。
SYN
后,两者状态都变为
SYN-REVD
。
ACK + SYN
,这个报文在对方接收之后,两者状态一起变为
ESTABLISHED
。
ESTABLISHED
状态。
FIN
报文,在 TCP 报文中的位置如下图:
FIN-WAIT-1
状态。注意, 这时候客户端同时也变成了
half-close(半关闭)
状态,即无法向服务端发送报文,只能接收。
CLOSED-WAIT
状态。
FIN-WAIT2
状态。
FIN
,自己进入
LAST-ACK
状态,
FIN
后,自己变成了
TIME-WAIT
状态,然后发送 ACK 给服务端。
MSL
(
Maximum Segment Lifetime,报文最大生存时间
), 在这段时间内如果客户端没有收到服务端的重发请求,那么表示 ACK 成功到达,挥手结束,否则客户端重发 ACK。
FIN
, 往往不会立即返回
FIN
, 必须等到服务端所有的报文都发送完毕了,才能发
FIN
。因此先发一个
ACK
表示已经收到客户端的
FIN
,延迟一段时间才发
FIN
。这就造成了四次挥手。
ACK
和
FIN
的发送合并为一次挥手,这个时候长时间的延迟可能会导致客户端误以为
FIN
没有到达客户端,从而让客户端不断的重发
FIN
。
CLOSED
变为
LISTEN
, 同时在内部创建了两个队列:
半连接队列和
全连接队列,即
SYN队列和
ACCEPT队列。
SYN
到服务端,服务端收到以后回复
ACK
和
SYN
,状态由
LISTEN
变为
SYN_RCVD
,此时这个连接就被推入了
SYN队列,也就是
半连接队列。
ACK
, 服务端接收后,三次握手完成。这个时候连接等待被具体的应用取走,在被取走之前,它会被推入另外一个 TCP 维护的队列,也就是
全连接队列(Accept Queue)。
SYN
。对于服务端而言,会产生两个危险的后果:
处理大量的SYN
包并返回对应ACK
, 势必有大量连接处于SYN_RCVD
状态,从而占满整个半连接队列,无法处理正常的请求。
由于是不存在的 IP,服务端长时间收不到客户端的ACK
,会导致服务端不断重发数据,直到耗尽服务端的资源。
SYN
后不立即分配连接资源,而是根据这个
SYN
计算出一个Cookie,连同第二次握手回复给客户端,在客户端回复
ACK
的时候带上这个
Cookie
值,服务端验证 Cookie 合法之后才分配连接资源。
四元组
——源 IP、源端口、目标 IP 和目标端口。
Sequence number
, 指的是本报文段第一个字节的序列号。
Initial Sequence Number(初始序列号)
,在三次握手的过程当中,双方会用过
SYN
报文来交换彼此的
ISN
。
ACK(Acknowledgment number)
。用来告知对方下一个期望接收的序列号,
小于ACK的所有字节已经全部收到。
SYN
,
ACK
,
FIN
,
RST
,
PSH
。
FIN
:即 Finish,表示发送方准备断开连接。
RST
:即 Reset,用来强制断开连接。
PSH
:即 Push, 告知对方这些数据包收到后应该马上交给上层的应用,不能缓存。
Cookie
, 用它同样可以实现 TFO。
SYN
给服务端,服务端接收到。
SYN Cookie
, 将这个
Cookie
放到 TCP 报文的
Fast Open
选项中,然后才给客户端返回。
Cookie
、
SYN
和
HTTP请求
(是的,你没看错)发送给服务端,服务端验证了 Cookie 的合法性,如果不合法直接丢弃;如果是合法的,那么就正常返回
SYN + ACK
。
ACK
还得正常传过来,不然怎么叫三次握手嘛。
timestamp
是 TCP 报文首部的一个可选项,一共占 10 个字节,格式如下:
kind(1 字节) + length(1 字节) + info(8 个字节)
timestamp
中存放的内容就是 a 主机发送时的内核时刻
ta1
。
timestamp
中存放的是 b 主机的时刻
tb
,
timestamp echo
字段为从 s1 报文中解析出来的 ta1。
ta1
, 也就是 s2 对应的报文最初的发送时刻。然后直接采用 ta2 - ta1 就得到了 RTT 的值。
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 ~ 2
的数据包了,怎么区分谁是谁呢?这个时候就产生了序列号回绕的问题。
SRTT = (α * SRTT) + ((1 - α) * RTT)
0.8
,范围是
0.8 ~ 0.9
。
RTO = min(ubound, max(lbound, β * SRTT))
1.3 ~ 2.0
,
lbound是下界,
ubound是上界。
0.8 ~ 0.9
, RTT 对于 RTO 的影响太小。
Jacobson / Karels 算法
。
SRTT
,公式如下:
SRTT = (1 - α) * SRTT + α * RTT
α
跟经典方法中的
α
取值不一样了,建议值是
1/8
,也就是
0.125
。
RTTVAR
(round-trip time variation)这个中间变量。
RTTVAR = (1 - β) * RTTVAR + β * (|RTT - SRTT|)
RTO
:
RTO = µ * SRTT + ∂ * RTTVAR
µ
建议值取
1
,
∂
建议值取
4
。
滑动窗口
的概念。
send
, WND 即
window
, UNA 即
unacknowledged
, 表示未被确认,NXT 即
next
, 表示下一个发送的位置。
receive
,NXT 表示下一个接收的位置,WND 表示接收窗口大小。
可用窗口
减少了 100 个字节,这很好理解。
60
个字节被留在了缓冲队列中。
拥塞控制
需要处理的问题。
接收端
给的限制
发送端
的限制
发送窗口
的大小。
发送窗口
?
发送窗口大小 = min(rwnd, cwnd)
cwnd
的变化。
慢启动
。运作过程如下:
cwnd
翻倍,现在
cwnd
只是增加 1 而已。
SACK
这个属性,通过
left edge
和
right edge
告知发送端已经收到了哪些区间的数据报。因此,即使第 5 个包丢包了,当收到第 6、7 个包之后,接收端依然会告诉发送端,这两个包到了。剩下第 5 个包没到,就重传这个包。这个过程也叫做
选择性重传(SACK,Selective Acknowledgment),它解决的是
如何重传的问题。
tcp_in_quickack_mode
设置)
keep-alive
, 不过 TCP 层面也是有
keep-alive
机制,而且跟应用层不太一样。
sudo sysctl -a | grep keepalive // 每隔 7200 s 检测一次 net.ipv4.tcp_keepalive_time = 7200 // 一次最多重传 9 个包 net.ipv4.tcp_keepalive_probes = 9 // 每个包的间隔重传间隔 75 s net.ipv4.tcp_keepalive_intvl = 75
keep-alive
选项,为什么?
The above is the detailed content of TCP协议灵魂 12 问,总会用得到. For more information, please follow other related articles on the PHP Chinese website!