【TCP】滑动窗口、流量控制 以及拥塞控制

2023-09-18 17:53:36

1. 滑动窗口(效率机制)

TCP 使用 确认应答 策略,对每一个发送的数据段,都要给一个 ACK 确认应答。收到 ACK 后再发送下一个数据段。这样做有一个比较大的缺点,就是性能较差。尤其是数据往返的时间较长的时候。

在这里插入图片描述

既然这样一发一收的方式性能较低,那么我们一次发送多条数据,使用滑动窗口,就可以大大的提高性能(其实是将多个段的等待时间重叠在一起了)。
滑动窗口存在的意义就是在保证可靠性的前提下,尽量提高效率。

在这里插入图片描述

  • 窗口大小指的是无需等待确认应答而可以继续发送数据的最大值。上图的窗口大小就是 4000 个字节(四个段)。
  • 发送前四个段的时候,不需要等待任何 ACK,直接发送
  • 收到第一个 ACK 后,滑动窗口向后移动,继续发送第五个段的数据;依次类推;
    (相当于一份等待时间等待多份 ACK, 当然不能不等,可靠传输的灵魂就是确认应答, 若没有 ACK,可靠传输就形同虚设。)
  • 操作系统内核为了维护这个滑动窗口,需要开辟 发送缓冲区 来记录当前还有哪些数据没有应答;只有确认应答过的数据,才能从缓冲区删掉;
  • 在一定范围内,窗口越大,传输速率就越快,网络的吞吐率就越高

在这里插入图片描述

那么如果出现了丢包,如何进行重传?这里分两种情况讨论。

  • 情况一:数据包已经抵达,ACK 被丢了。

在这里插入图片描述
这种情况下,部分 ACK 丢了并不要紧,因为可以通过后续的ACK进行确认.
ACK 的确认号有特定含义,保证后一条 ACK 覆盖前一条,
比如并没有收到 1001 ACK, 但是收到了 2001 ACK 就说明 2001 之前的数据全部已经收到了
若发送 4001 ~ 5000 之前,只收到了 4001, 但是它的意思是,4001 之前的都收到了,窗口就可以一次往下挪动 4 个。

  • 情况二:数据包就直接丢了

在这里插入图片描述

  • 当某一段报文段丢失之后,发送端会一直收到 1001 这样的 ACK,就像是在提醒发送端 “我想要的是 1001” 一样;
  • 如果发送端主机连续三次收到了同样一个 “1001” 这样的应答,就会将对应的数据 1001 - 2000 重新发送;
  • 这个时候接收端收到了 1001 之后,再次返回的 ACK 就是 7001 了(因为2001 - 7000)接收端其实之前就已经收到了,被放到了接收端操作系统内核的接收缓冲区中;
    重传只需要把丢的数据重传就行了,后面已经传过的数据不用再传了。

这种机制被称为 “高速重发控制”(也叫 “快重传”)。
(为什么说是 “快” 重传,因为可能收到三个连续相同的 ACK 的时间内还没有触发超时机制,也就是还没超时呢,但是不等触发超时,直接就重传了。)

2. 流量控制(安全机制)

流量控制是滑动窗口的延伸,目的是为了保证可靠性。

  • 在一定范围内,滑动窗口越大传输效率就越高,但是不能只考虑发送方,不考虑接收方,
  • 接收端处理数据的速度是有限的。如果发送端发的太快,导致接收端的缓冲区被打满,这个时候如果发送端继续发送,就会造成丢包,继而引起丢包重传等等一系列连锁反应。

因此 TCP 支持根据接收端的处理能力,来决定发送端的发送速度。这个机制就叫做流量控制(Flow Control);

  • 接收端将自己可以接收的缓冲区大小放入 TCP 首部中的 “窗口大小” 字段,通过 ACK 端通知发送端;
  • 发送方接收到这个这个数据后,就会灵活的调整发送速度,调整窗口大小
  • 窗口大小字段越大,说明网络的吞吐量越高;
  • 接收端一旦发现自己的缓冲区快满了,就会将窗口大小设置成一个更小的值通知给发送端;
  • 发送端接受到这个窗口之后,就会减慢自己的发送速度
  • 如果接收端缓冲区满了,就会将窗口置为 0;
  • 这时发送方不再发送数据,但是需要定期发送一个窗口探测数据段,使接收端把窗口大小告诉发送端。

在这里插入图片描述

  • 接收端如何把窗口大小告诉发送端呢?
    回忆我们的 TCP 首部中,有一个16 位窗口字段,就是存放了窗口大小信息;

  • 那么问题来了,16 位数字最大表示 65535,那么 TCP 窗口最大就是 65535 字节么?
    实际上,TCP 首部40字节选项中还包含了一个窗口扩大因子 M,实际窗口大小是 窗口字段的值左移 M 位即 65535 * 2 ^M

3. 拥塞控制(安全机制)

拥塞控制,也是滑动窗口的延伸,限制滑动窗口的发送速率。
拥塞控制描述的是发送方到接收方整个链路直接的拥堵情况。

  • 最终的滑动窗口的大小 = Min (流量控制窗口,拥塞控制窗口)

虽然TCP有了滑动窗口这个大杀器,能够高效可靠的发送大量的数据。
但是如果在刚开始阶段就发送大量的数据,仍然可能引发问题。
因为网络上有很多的计算机,可能当前的网络状态就已经比较拥堵。在不清楚当前网络状态下,贸然发送大量的数据,是很有可能引起雪上加霜的。

  • 发送方开始时以一个较小的窗口来发送数据,
  • 若数据很流畅到达,就逐渐加大窗口大小,
  • 若加大到一定程度出现丢包,就减小窗口,
  • 通过反复的增大/减小过程,逐渐找到一个合适的范围,拥塞窗口就在这个范围中不断变化,达到一个 “动态平衡”。

具体拥塞窗口是怎么变化的呢 ?

在这里插入图片描述

  1. 慢开始
    初始值 窗口大小为 1, 然后以指数级别增长,“慢开始” 只是指初使时慢,但是增长速度非常快。
  2. 拥塞避免
    窗口值到达 ssthresh 时,从指数增长变为 线性增长。
  3. 网络拥塞
    出现大量丢包情况,说明网络拥塞了,拥塞窗口大小直接变为 1 。ssthresh 阈值变为此时拥塞窗口大小的一半,图中就是从 变为 24 的一半 12 。
    (少量的丢包,我们仅仅是触发超时重传;大量的丢包,我们就认为网络拥塞;)
  4. 然后重新慢开始,循环这个过程。

当TCP通信开始后,网络吞吐量会逐渐上升;随着网络发生拥堵,吞吐量会立刻下降;

  • 为什么使用指数级别的增长速度 ?
    因为希望能快速接近 ssthresh 阈值, 既希望速度快,又希望不大量丢包,如果初始情况给的窗口大小很小,可能合适的值是个很大的值,那么使用指数增长的话,能够很快的接近这个值。

  • ssthresh 的意义 ?
    决定了什么时候从指数增长变为线性增长

  • 拥塞窗口最理想的大小 ?
    ssthresh 值与 出现拥塞的这个值之间是最理想的效果,这个范围之间传输速率较快,并且没有大量丢包。

  • 为什么出现拥塞时,直接让窗口变为初始值 1 ?
    因为网络的情况很复杂,不稳定,如果出现大量丢包,很可能速度降下来一点是不能解决问题的,速度降得太慢还会有可能出现持续丢包,就会对网络质量带来很大影响,一下让窗口变得很小,就是期望这次传输一定能成功。

拥塞控制,归根结底是TCP协议想尽可能快的把数据传输给对方,但是又要避免给网络造成太大压力的折中方案。

常见面试题:

  1. UDP 本身是无连接,不可靠,面向数据报的协议,如果要基于传输层UDP协议,来实现一个可靠传输,应该如何设计?
  2. UDP 大小是受限的,如果要基于传输层UDP协议,传输超过64K的数据,应该如何设计?

以上两个问题答案类似,都可以参考TCP的可靠性机制然后在应用层实现类似的逻辑:
如:

  • 引入序列号,保证数据顺序;
  • 引入确认应答,确保对端收到了数据,保证可靠传输;
  • 引入超时重传,如果隔一段时间没有应答,就重发数据;
  • 引入滑动窗口;
  • 引入窗口扩大选项;
  • 引入流量控制
  • 引入拥塞控制
  • ……

好啦! 以上就是对 TCP 滑动窗口、流量控制 以及拥塞控制的讲解,希望能帮到你 !
评论区欢迎指正 !

更多推荐

大型集团借力泛微搭建语言汇率时区统一、业务协同的国际化OA系统

国际化、全球化集团,业务遍布全世界,下属公司众多,集团对管理方式和企业文化塑造有着很高的要求。不少大型集团以数字化方式助力全球统一办公,深化企业统一管理。面对大型集团全球化的管理诉求,数字化办公系统作为集团日常使用的平台,自然需要适应企业管理升级,灵活适应国际化带来的管理发展变化和满足全球协同办公需求。最重要的是实现信

华为OD:跳房子I

题目描述跳房子,也叫跳飞机,是一种世界性的儿童游戏。游戏参与者需要分多个回合按顺序跳到第1格直到房子的最后一格跳房子的过程中,可以向前跳,也可以向后跳。假设房子的总格数是count,小红每回合可能连续跳的步教都放在数组steps中,请问数组中是否有一种步数的组合,可以让小红两个回合跳到最后一格?如果有,请输出索引和最小

ffmpeg抠图

1.不用png,用AVFrame2.合流3.图片抠图透明(1.)mp4扣yuv图,(2)用1.把一张yuv标记为透明然后av_hwframe_transfer_data到GPU(3)用抠图算法函数对yuv进行处理(4)qsv的h264_qsv只支持nv12和qsv,但qsv本身并不限制像素格式,比如在qsv里可以用vp

MES系统是如何采集Modbus设备数据的呢?

随着工业自动化的不断发展,各种协议和标准在行业中变得越来越重要。其中,Modbus协议是一种在工业自动化领域非常流行的通信协议,而Modbus网关是实现Modbus协议转换的关键设备。一、MES系统与Modbus网关MES系统即制造执行系统(ManufacturingExecutionSystem),是一套面向制造过程

【含java面试题】深入解析栈溢出及JVM参数设置

AI绘画关于SD,MJ,GPT,SDXL百科全书面试题分享点我直达2023Python面试题2023最新面试合集链接2023大厂面试题PDF面试题PDF版本java、python面试题项目实战:AI文本OCR识别最佳实践AIGamma一键生成PPT工具直达链接玩转cloudStudio在线编码神器玩转GPUAI绘画、A

SQL性能优化

当你的数据里只有几千几万,那么SQL优化并不会发挥太大价值,但当你的数据里去到了几百上千万,SQL优化的价值就体现出来了!因此稍微有些经验的同学都知道,怎么让MySQL查询语句又快又好是一件很重要的事情。要让SQL又快又好的前提是,我们知道它「病」在哪里,而explain关键字就是MySQL提供给我们的一把武器!在我们

需要每日温习的java八股文基础

一、java(1)集合1.list:LinkedList、ArrayList和VectorArrayList、LinkedList和Vector的区别,场景和扩容机制源码LinkedList接口实现类,链表,插入删除,没有同步,线程不安全ArrayList接口实现类,数组,随机访问,没有同步,线程不安全Vector接口

SpringBoot_快速入门

一、SpringBoot概述文档下载地址:https://docs.spring.io/spring-boot/docs/current/reference/1.SpringBoot介绍概述:SpringBoot开发团队认为Spring操作太繁琐了,目的在于简化开发配置,能够快速搭建开发环境,集成了tomcat,对于开

如何使用反 CSRF 令牌保护您的网站和 Web 应用程序

防止跨站点请求伪造攻击(CSRF/XSRF)的最常见方法是使用反CSRF令牌,该令牌只是一个唯一值集,然后由Web应用程序需要。CSRF是一种客户端攻击,可用于将用户重定向到恶意网站、窃取敏感信息或在用户会话中执行其他操作。幸运的是,使用CSRF令牌来保护用户免受CSRF攻击及其后果相对容易。CSRF原理那我们具体看看

前端工程师笔试题【校招】

1.设一个有序的单链表中有n个结点,现要求插入一个新结点后使得单链表仍然保持有序,则该操作的时间复杂度为()。如果要在一个有序的单链表中插入一个新节点,以保持单链表的有序性,时间复杂度取决于插入的位置。以下是几种常见的情况:在链表的头部插入新节点:如果要在链表的头部插入一个新节点,只需要将新节点的指针指向当前链表的头节

3、ARM寄存器组织

ARM寄存器组织1、寄存器概念寄存器是处理器内部的存储器,没有地址作用一般用于暂时存放参与运算的数据和运算的结果分类包括通用寄存器、专用寄存器、控制寄存器2、ARM寄存器注:在某个特定的模式下,只能使用当前模式下的寄存器。一个模式下的特有寄存器在其他模式下不可以使用专用寄存器R15(PC,ProgramCounter)

热文推荐