当前位置: 首页 > news >正文

TCP ---滑动窗口以及拥塞窗口

序言

 在上一篇文章中我们介绍了 TCP 中的协议段格式,以及保证其可靠传输的重传机制,着重介绍了三次握手建立连接,四次挥手断开连接的过程(👉点击查看)。
 这只是 TCP 保证通信可信策略的一部分,现在让我们继续深入吧!


流量控制

1. 流量控制的目的

 一个报文千辛万苦现在到达了目的主机,但是目的主机的缓冲区已经满了,那只有对不起了,只能把你丢弃了!对于 TCP 协议来说,我的报文已经发送了但是迟迟不见回应,那是不是丢了呀?我再发!另一方说,实在没位置了,你发一个我只有丢一个!网络资源就这样白白的浪费了,这种情况应该避免!
 知道了问题,我们就应该思考问题的本质,为什么会这样?怎样避免?归根到底就是 发送方不知道接收方的接受能力导致的!如果发送方能知道接收方的接受能力从而按照对方的实际接受能力发送的话就不会出现这种问题! 这就是流量控制的目的。

2. 滑动窗口

 在 TCP 协议段中有一个字段叫做 16位窗口大小,这就是在告诉对方我现在的可以接受的空间是多少,但是只有这个显然是不够的。咋们首先看一下滑动窗口到底是什么:
在这里插入图片描述
好了,本来对滑动窗口不熟悉的我,看了这幅图就更懵了😥!

 首先我们需要理解 滑动窗口的窗口中维护的是需要发送的数据,因为窗口大小是和对方的接受能力相关的! 现在我想要提一个问题,比如我现在发送了 22 这个报文,那么我还得需要维护 22 嘛,还是窗口向右滑动呢?答案是,还需要维护 22 ,只有收到了相应的应答报文才可以向右滑动。

 起始从简单理解的话,滑动窗口无非就是两个指针的移动,为了更好地理解,咋们来模拟一遍滑动的过程:

  1. 左指针为 22(代表窗口左边界),窗口大小为 8 字节,右指针为 29 (代表窗口右边界),现在将窗口的所有值发送
  2. 收到 22 的应答报文,返回的确认序号为 23, 代表下一次的起始,返回的窗口大小为 7。现在左指针指向 23,右指针指向 29
  3. 直接收到 26 的应答报文,返回的确认序号 27,下一次的开始,返回的窗口大小为 3。现在左指针指向 27,右指针指向 29。(💡知识回顾:为什么收到 26 的应答报文就直接更新了呢,前面的 23,24,25不管了吗?这是因为应答报文的定义,收到该应答报文,就代表着该报文之前的数据也已经收到!
  4. 收到 28 的应答报文,返回确认序号29,下一次的开始,返回的窗口大小为 5。现在左指针指向 29,右指针指向 33。将新加入窗口的数据发送(30 - 33)!

上述没有发生丢包的异常情况,如果有也别担心,会触发重传机制(快重传 or 超时重传)。

3. 窗口关闭问题

 滑动窗口实现了发送方按需发送的能力,但是还有一个问题。有没有可能出现接收方接收数据后,应用层处理数据的速度太慢了导致缓冲区数据满了,造成窗口大小最后返回 0 ,也就是窗口关闭的问题?
 对于这个问题,接收方和发送方都有一定的策略:

  • 对于接收方:TCP 为每个连接设有一个持续定时器,只要 TCP 连接一方收到对方的零窗口通知,就启动持续计时器。如果持续计时器超时,就会发送 窗口探测(Window probe)报文,而对方在确认这个探测报文时,给出自己现在的接收窗口大小。当然,如果多次探测之后窗口的大小还是 0 的话,就会发送 RST 中断连接。
  • 对于接收方:处理完数据后窗口自然而然增大了,于是会向接收方发送窗口大小。

拥塞控制

1. 拥塞控制的目的

 有了滑动窗口就极大避免了发送接收能力不对等的问题,但是正常的发送接收数据还和网络的状态有极大的状态。就比如早上四五点的网络状态肯定是比起晚上七八点要好很多的。当网络拥塞时,我们发送的数据包极有可能产生时延以及丢包问题,这会导致我们再次重发,加剧网络拥塞的状态。
 拥塞的问题产生的原因就是网络上的数据包太多了,负担太大了。TCP 为了避免这种情况,使用了 拥塞窗口

2. 慢启动

 了解该算法之前,我们首先得知道拥塞窗口是什么:

  • 发送开始的时候,定义拥塞窗口大小为 1
  • 每次收到一个 ACK 应答, 拥塞窗口加 1
  • 每次发送数据包的时候,将拥塞窗口和接收端主机反馈的窗口大小做比较, 取较 小的值作为实际发送的窗口

现在了解了拥塞窗口,我们就可以了解慢启动算法的具体过程了:
在这里插入图片描述

  1. 发生拥塞时,拥塞窗口的值变为 1 ,并且呈指数级别增长
  2. 当到达一个门限时,由指数增长变为线性增长
  3. 再次发生拥塞时,拥塞窗口变为 1 ,并且更新门限(新的门限 = 发生拥塞时的窗口大小 / 2
  4. 重复 1- 3 的过程

了解了大概的过程,我们肯定心里也已经有了疑惑:

  • 为什么要将窗口变为 1?并且采用指数增长?
  • 为什么又要采用线性增长?
  • 门限窗口的更新为什么是 发生拥塞时的窗口大小 / 2

问题一:
 发生拥塞时,在网络上的负担已经很重了,我们刚开始发送少数量的数据先让网络上的数据疏通之后,在慢慢增加发送数据的量恢复正常的通信速度。慢启动只是 刚开始发送的数据少,但是数据增加的速度可一点也不慢!

问题二:
 如果一直采用指数增长的模式,那发送的数据量就会非常大,不出意外会快速的造成下一次拥塞的发生!所以替换为指数增长。有人肯定会想不增长行吗?一直维护在一个不会发生拥塞的大小。
网络的情况是一直在变化的 ! 正是因为如此,才需要不断地增长去探测网络的承受能力。就比如七八点网络高峰期,拥塞发生的频繁,现在大家都维持在一个恰当的窗口大小来避免拥塞的发生。但是,可能过了七八点到了凌晨的时候,网络情况大大的缓和了,但是我们的并没有继续线性探测来试探网络状态,你还是以那个值作为窗口大小不就是浪费你的传输速率吗?

问题三:
 在这里的介绍慢启动的很多,解释明白了的很少。在这里我尝试用自己的理解来解释一下:网络是变化的,每一次发生拥塞时的窗口值都可能不一样,但是前后相邻两次的差距可能不会很大,所以当指数增长到上一次发生拥塞的值的一半时,就不要指数增长了,不然很可能再次拥塞,让用户的网速像过山车一样,一上一下一上一下。所以采取线性增长,不断地试探新的拥塞触发点,还可以让用户的网络情况尽量平稳一些。


面向字节流

1. 什么是面向字节流

 我们在前面一直在提到 TCP 面向字节流,UDP 是面向数据报。那么到底是什么个意思呢?
 首先我们先了解 面向数据报,当我们使用 UDP 发送数据时,该数据在传输层只会加一个报头然后交给下一层处理,对方在 UDP 所读取的也是一个完整的报文然后交付给上一层。总而言之,一个报文就是一次完整的发送数据
TCP 是有一个真正意义上的缓冲区的,当我们需要发送数据时,会先将数据暂存在缓冲区,至于多久发送取决 发送窗口、拥塞窗口以及当前发送缓冲区的大小等条件。就比如我想要发送两条信息:Nihao! and How is going? 在底层的发送顺序可能是:

  • Nihao | How is going?
  • Nihao How is gong?
  • Ni | hao How is gong?

所以,我们不能简单认为 一个信息对应一个报文。面向字节流和面向数据报之间最大的区别就是 报文和报文之间有没有明确的边界。

2. 粘包问题

 因为 TCP 是面向字节流的,所以不可避免的报文和报文之间没有明显地边界,那该怎么办呢?TCP 并不负责这个,所以需要我们在应用层来处理。

1. 定长消息

 我们约定好我们一次传输报文的大小是多少字节,然后只有接收方每次接收到指定字节大小数据,那么就认为读取到了一次完整的报文!但是这种方法不灵活。

2. 分隔符协议

 发送方在每个数据包的结尾添加一个特定的分隔符,接收方根据此分隔符进行拆包操作。

还有许多方法在这里就不介绍了。


总结

 这篇文章中,我们主要介绍了使用滑动窗口来控制流量,拥塞窗口解决拥塞问题,最后介绍了面向字节流,希望大家有所收获!

相关文章:

TCP ---滑动窗口以及拥塞窗口

序言 在上一篇文章中我们介绍了 TCP 中的协议段格式,以及保证其可靠传输的重传机制,着重介绍了三次握手建立连接,四次挥手断开连接的过程(👉点击查看)。  这只是 TCP 保证通信可信策略的一部分,现在让我们继续深入吧&…...

第十二章--- fixed 和 setprecision 函数、round 函数、进制转换及底层逻辑

1. 保留几位小数 在C中&#xff0c;如果你想要控制输出的小数点后的位数&#xff0c;可以使用<iomanip>头文件提供的fixed和setprecision函数。这里的fixed用于设置浮点数的输出格式为定点表示法&#xff0c;而setprecision(n)则用来指定小数点后保留的位数。具体用法如…...

ASP.NetCore---I18n(internationalization)多语言版本的应用

文章目录 0.实现的效果如下1.创建新项目I18nBaseDemo2.添加页面中的下拉框3.在HomeController中添加ChangeLanguage方法4.在Progress.cs 文件中添加如下代码&#xff1a;5. 在progress.cs中添加code6.添加Resource资源文件7.在页面中引用i18n的变量8. 重启项目&#xff0c;应该…...

vue3 环境配置vue-i8n国际化

一.依赖和插件的安装 主要是vue-i18n和 vscode的自动化插件i18n Ally https://vue-i18n.intlify.dev/ npm install vue-i18n10 pnpm add vue-i18n10 yarn add vue-i18n10 vscode在应用商城中搜索i18n Ally&#xff1a;如图 二.实操 安装完以后在对应项目中的跟package.jso…...

2024 uniapp入门教程 01:含有vue3基础 我的第一个uniapp页面

uni-app官网uni-app,uniCloud,serverless,快速体验,看视频&#xff0c;10分钟了解uni-app,为什么要选择uni-app&#xff1f;,功能框架图,一套代码&#xff0c;运行到多个平台https://uniapp.dcloud.net.cn/ 准备工作&#xff1a;HBuilder X 软件 HBuilder X 官网下载&#xf…...

CentOS 7文件系统

从centos7开始&#xff0c;默认的文件系统从ext4变成了XFS。随着虚拟化的应用越来越广泛&#xff0c;作为虚拟化磁盘来源的大文件&#xff08;单个文件几GB级别&#xff09;越来越常见。 1.XFS组成部分&#xff1a; XFS文件系统在数据的分布上主要划分为三部分&#xff1a;数据…...

vue源码解析(源码解析学习大纲)

文章目录 Vue源码解析入手方向大纲1.核心概念1-1.响应式系统1-2. 组件1-3. 虚拟DOM1-4. 指令1-5. 生命周期钩子 2.虚拟DOM2-1. 概念2-2. 工作流程2-3. 示例2-4.总结 3.组件系统3-1. 组件的定义3-2. 组件的创建3-3. 组件的模板3-4. 生命周期3-5. 事件处理3-6. 插槽&#xff08;S…...

工行企业网银U盾展期后有两个证书问题的解决方法

工行企业网银U盾证书快到期后&#xff0c;可以自助展期&#xff0c;流程可以根据企业网银提示页面操作。操作后&#xff0c;可能存在两个新旧两个证书并存的情况&#xff0c;致使网银转账等操作失败&#xff0c;如图&#xff1a; 其原因是新证书生成后&#xff0c;旧证书没有删…...

《Linux从小白到高手》理论篇:文件权限控制及文件操作相关的命令

List item 本篇介绍Linux文件权限控制及文件操作相关的命令&#xff0c;看完本文&#xff0c;有关Linux文件权限控制及文件操作相关的常用命令你就掌握了99%了。 文件权限 在介绍文件权限之前先来复习下Linux的文件类型&#xff0c;始终记住那句话&#xff1a;Linux系统下&a…...

前端框架React的详细的学习方法和过程

学习React作为前端架构的一部分&#xff0c;是一个系统且逐步深入的过程。以下是一个详细的学习方法和过程&#xff0c;可以帮助你有效地掌握React&#xff1a; 1. 理解React的基础知识 首先&#xff0c;你需要了解React的基本概念&#xff0c;包括它是什么、为什么使用它以及…...

linux中缓存,在kafka上应用总结

linux中的缓存 页缓存 pagecatch&#xff08;读缓存用于提供快速读&#xff09;块缓存&#xff08;用于提供其他设备快速写&#xff09;当对读缓存读的时候&#xff0c;修改了读的数据&#xff0c;页缓存就会被标记为脏数据&#xff0c;等到写的时候它会向块缓存同步数据&…...

前端练习小项目 —— 让图片变得更 “色”

前言&#xff1a;相信读者在学习完了HTML、CSS和JavaScript之后已经想要迫不及待的想找一个小型的项目来练练手&#xff0c;那么这篇文章就正好能满足你的 “需求”。 ✨✨✨这里是秋刀鱼不做梦的BLOG ✨✨✨想要了解更多内容可以访问我的主页秋刀鱼不做梦-CSDN博客 在开始学习…...

时间卷积网络(TCN)原理+代码详解

目录 一、TCN原理1.1 因果卷积&#xff08;Causal Convolution&#xff09;1.2 扩张卷积&#xff08;Dilated Convolution&#xff09; 二、代码实现2.1 Chomp1d 模块2.2 TemporalBlock 模块2.3 TemporalConvNet 模块2.4 完整代码示例 参考文献 在理解 TCN 的原理之前&#xff…...

零散的知识

1.物化 在SQL中&#xff0c;物化&#xff08;Materialization&#xff09;是指将查询结果保存为物理数据结构以供后续使用的过程。这与普通的视图或查询不同&#xff0c;物化视图会存储查询的结果&#xff0c;而不是每次查询时都动态地重新计算数据。 ①物化视图 物化视图是一…...

Python读取pdf中的文字与表格

一、PyPDF2包安装 在Python中安装PyPDF2库&#xff0c;您可以使用pip包管理器。打开您的命令行工具&#xff08;例如CMD、Terminal或Anaconda Prompt&#xff09;&#xff0c;然后输入以下命令&#xff1a; pip install PyPDF2 如果您使用的是Python 3&#xff0c;并且系统中…...

【MySQL 08】复合查询

目录 1.准备工作 2.多表查询 笛卡尔积 多表查询案例 3. 自连接 4.子查询 1.单行子查询 2.多行子查询 3.多列子查询 4.在from子句中使用子查询 5.合并查询 1.union 2.union all 1.准备工作 如下三个表&#xff0c;将作为示例&#xff0c;理解复合查询 EMP员工表…...

求1000以内的完数

题目&#xff1a;一个数如果恰好等于他的因子之和&#xff08;包括1&#xff0c;但不包括这个数&#xff09;&#xff0c;这个数就是完数。编写算法找出1000之内的所有完数&#xff0c;并按下面格式输出其因子&#xff1a;28 its factors are 1,2,4,7,14 代码如下&#xff1a;…...

sqli-labs less-16 post提交dnslog注入

post提交DNSlog注入 第十六关和和十五关大差不大&#xff0c;可以使用布尔注入&#xff0c;时间盲注等&#xff0c;只不过闭合方式不一样&#xff0c;但是用布尔和时间盲太过于消耗时间&#xff0c;本次测试我将使用dnslog注入。 使用在线平台http://www.dnslog.cn/ 闭合方式…...

nginx报错|xquic|xqc_engine_create: fail|

一.问题描述 nginx使用xquic协议一切安装正常,nginx -s reload也正常&#xff0c;但就是访问不了网页 [emerg] 12342#0: |xquic|xqc_engine_create: fail| [emerg] 12342#0: |xquic|ngx_xquic_process_init|engine_init fail| [emerg] 12341#0: |xquic|xqc_engine_create: fai…...

Java虚拟机(JVM)

目录 内存区域划分堆&#xff08;Heap&#xff09;方法区&#xff08;Method Area&#xff09;程序计数器&#xff08;Program Counter Register&#xff09;虚拟机栈&#xff08;VM Stack&#xff09;本地方法栈&#xff08;Native Method Stack&#xff09; 类加载的过程类加…...

[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?

&#x1f9e0; 智能合约中的数据是如何在区块链中保持一致的&#xff1f; 为什么所有区块链节点都能得出相同结果&#xff1f;合约调用这么复杂&#xff0c;状态真能保持一致吗&#xff1f;本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里&#xf…...

网络六边形受到攻击

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 抽象 现代智能交通系统 &#xff08;ITS&#xff09; 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 &#xff08;…...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

系统设计 --- MongoDB亿级数据查询优化策略

系统设计 --- MongoDB亿级数据查询分表策略 背景Solution --- 分表 背景 使用audit log实现Audi Trail功能 Audit Trail范围: 六个月数据量: 每秒5-7条audi log&#xff0c;共计7千万 – 1亿条数据需要实现全文检索按照时间倒序因为license问题&#xff0c;不能使用ELK只能使用…...

Nginx server_name 配置说明

Nginx 是一个高性能的反向代理和负载均衡服务器&#xff0c;其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机&#xff08;Virtual Host&#xff09;。 1. 简介 Nginx 使用 server_name 指令来确定…...

【Zephyr 系列 10】实战项目:打造一个蓝牙传感器终端 + 网关系统(完整架构与全栈实现)

🧠关键词:Zephyr、BLE、终端、网关、广播、连接、传感器、数据采集、低功耗、系统集成 📌目标读者:希望基于 Zephyr 构建 BLE 系统架构、实现终端与网关协作、具备产品交付能力的开发者 📊篇幅字数:约 5200 字 ✨ 项目总览 在物联网实际项目中,**“终端 + 网关”**是…...

相机Camera日志分析之三十一:高通Camx HAL十种流程基础分析关键字汇总(后续持续更新中)

【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:有对最普通的场景进行各个日志注释讲解,但相机场景太多,日志差异也巨大。后面将展示各种场景下的日志。 通过notepad++打开场景下的日志,通过下列分类关键字搜索,即可清晰的分析不同场景的相机运行流程差异…...

leetcodeSQL解题:3564. 季节性销售分析

leetcodeSQL解题&#xff1a;3564. 季节性销售分析 题目&#xff1a; 表&#xff1a;sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...

大模型多显卡多服务器并行计算方法与实践指南

一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

Unit 1 深度强化学习简介

Deep RL Course ——Unit 1 Introduction 从理论和实践层面深入学习深度强化学习。学会使用知名的深度强化学习库&#xff0c;例如 Stable Baselines3、RL Baselines3 Zoo、Sample Factory 和 CleanRL。在独特的环境中训练智能体&#xff0c;比如 SnowballFight、Huggy the Do…...