赞
踩
x
的同步报文,进入 SYN-SENT
状态。y
的同步确认报文,进入 SYN-RECEIVED
状态。ESTABLISHED
状态,此时成功建立长连接。CLOSE-WAIT
状态,此时表明客户端到服务端的连接已经释放,不再接受客户端的数据。但因为 TCP 是全双工的,所以服务器仍可以发送数据。LAST-ACK
状态。确认报文
,此时客户端进入 TIME-WAIT
状态,会等待 2MSL(最长报文段寿命),若期间没有收到服务器端的数据报文,进入 CLOSED
状态。服务器端在收到确认应答后也进入 CLOSED
态。两次不安全,四次没必要。
TCP 通信需要确保双方都具有数据收发的能力。
第一次和第二次不能携带数据,第三次可以携带数据。
握手失败有两种情况:
ISN 全称是 Initial Sequence Number,是 TCP 发送方数据编号的起点。
在 TCP 连接建立时,双方会交换各自的 ISN,并根据 ISN 来确定数据包的正确顺序。这样可以防止数据包被重复发送或重复接收,同时还可以防止恶意攻击者通过猜测ISN来进行连接劫持或伪造数据包。
TCP 每次建立连接时,初始化序列号都是随机的,这样一方面可以防止恶意攻击者通过猜测 ISN 来进行连接劫持或伪造数据包,另一方面可以防止在网络拥塞等情况下历史报文被下一个相同的四元组(源目套接字)接收。因此 ISN 一般都基于随机算法生成,同时还会整合时间戳等信息以避免序号回绕。
服务器第一次收到客户端的 SYN 之后会处于 SYN_RCVD
状态,此时双方还没有完全建立连接,服务器会把这种状态下的连接放在半连接队列(syn 队列)中,半连接队列可以在一定程度上抵御 syn 泛洪攻击。
而全连接队列(accept 队列)则保存着已经完成三次握手,但是还未被 accept()
取走的连接队列。
半连接队列和全连接队列都有最大长度限制,且二者长度限制都受到 lisnten()
系统调用中的 backlog
参数大小影响。超出限制时内核会直接丢弃收到的报文或返回一个 RST 包,其中直接丢弃可以更好地应对突发流量。
TCP 握手时服务端将 SYN 和 ACK 合并在一个包中发送,因此减少了一次握手。
对于四次挥手,由于 TCP 是全双工通信,客户端(或者说主动关闭方)发送 FIN 请求只能表示客户端不再发送数据了,不代表完全断开连接,服务端(或者说被动关闭方)可能还要发送数据。所以不能将服务端的 FIN 包和对客户端的 ACK 包合并发送,只能先确认主动关闭方的 FIN,等服务端数据发送完毕时再发送 FIN 包,故挥手需要四次。
假设主动关闭方最后发送的 ACK 在网络中丢失,由于 TCP 协议的重传机制,被动关闭方将会重发其 FIN,在该 FIN 到达主动关闭方之前,主动关闭方必须维护该连接(即该 TCP 连接所对应的套接字等资源不能被释放或重新分配)。直到客户端收到服务器重发的 FIN 之后重发 ACK,再经过 2MSL(客户端 ACK 的最大报文生存时间 + 服务器 FIN 最大报文生存时间)没有再收到新的 FIN 之后,该 TCP 连接才能恢复 CLOSED
状态。
如果主动关闭方不维护这样一个 TIME_WAIT
状态,那么当被动关闭方重发的 FIN 到达时,主动关闭方会用 RST 包响应对方,使得正常的关闭连接过程变为异常。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。