TCP协议之三次握手四次挥手

TCP协议,全称:传输控制协议。

1.TCP协议段格式:
在这里插入图片描述

  • 16位源端口号与目的端口号:确保端与端之间的数据传输
  • 32位序号与确认序号:确保数据的有序交付
  • 4位头部长度:解析时获取4位头部长度,最小为20字节,最大为60字节
  • 6位标志位:URG(紧急指针是否有效)、ACK(确认信号是否有效)、PSH(提示接收端应用程序立刻从TCP缓冲区读走数据)、RST(复位报文段)、SYN(同步报文段)、FIN(结束报文段)。
  • 16位窗口大小:实现滑动窗口以及进行流量控制
  • 16位校验和:保证数据的一致性
  • 16位紧急指针:带外数据
  • 40字节的选项数据:

2.TCP协议的连接管理机制:
在正常情况下,TCP要进行三次握手建立连接,四次挥手断开连接。
在这里插入图片描述
我们来梳理一下这个过程:
1)三次握手:

  • 首先,客户端调用connect(),向服务器端发送SYN请求连接,状态变为SYN_SENT;
  • 服务器端监听到SYN连接请求,并向客户端发送ACK确认报文以及SYN连接报文,状态变为SYN_RECV;
  • 客户端收到服务器的确认报文ACK,并回复给服务器端一个确认报文ACK,此时状态变为ESTABLISHED就绪状态;
  • 服务器端收到客户端的确认报文ACK,进入ESTABLISHED就绪状态。

现在成功建立连接,可以进行收发数据了。

2)四次挥手:

  • 首先,客户端调用close(),主动请求关闭连接,发送结束报文段FIN给服务器端,状态变为FIN_WAIT1;
  • 然后,服务器收到客户端的断开请求,并回复给客户端一个ACK,服务器状态变为CLOSE_WAIT状态;客户端状态变为FIN_WAIT2状态;
  • 等待服务器端完成了剩余工作后,向客户端发送结束报文FIN请求断开,此时服务器处于LASK_ACK状态,等待最后一次确认信号;
  • 客户端收到了服务器发来的FIN包,并回复给服务器端一个ACK确认断开,此时客户端处于TIME_WAIT状态;
  • 服务器端收到了最后一个ACK,状态变为CLOSED;
  • 客户端等待2MSL后,状态变为CLOSED状态,关闭连接。

3.针对这个连接关闭过程,我们有以下几个问题需要讨论:

  • 为什么握手是三次而挥手是四次呢?
    1)握手三次是为了保证通信双方都有收发数据的能力。
    两次握手不安全,如果只有两次握手,且SYN有延迟,每当有一个SYN来连接,就会为它新建一个socket;并且不能保证两方都有收发数据的能力。
    四次握手没必要,三次握手已经确认了数据双方具有收发数据的能力,不需要再进行确认。
    2)由于TCP协议是全双工的,所以需要双方都进行关闭,ACK与FIN包不能放在一起发送,所以需要四次挥手才能断开连接。

  • 若是三次握手失败,服务器会如何处理?
    服务端超时等待确认报文ACK后,会向客户端发送RST重置请求,然后释放新建的socket资源。

  • 主动关闭方TIME_WAIT的作用?
    在主动关闭方发送完最后一次ACK时,会等待2MSL时间,然后进入CLOSED状态,此时连接才真正关闭了。

    1)TIME_WAIT其实就避免了由于最后一次ACK丢失而对后续的新连接造成影响。
    假设没有TIME_WAIT而直接进入CLOSED状态,关释放了socket资源;则重启主动关闭方,再次进行连接时,会收到被动关闭方发来的FIN包,影响连接;主动关闭方发送SYN请求,但被动关闭方等待的是ACK,因为状态不对导致影响连接。
    可见,TIME_WAIT保证了被动关闭方有足够的时间来接受最后一次ACK;并保证了旧的报文消失在网络中,不会对新连接产生影响。

    2)为什么需要等待2MSL呢?
    MSL:报文最大生命周期
    等待2MSL是为了若对方重新发送FIN包,可以再次对其进行回复。
    这样保证了由于网络问题延迟的报文(如后续重传的ACK与FIN包)都消失在网络中,不会对后续连接造成影响。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值