网络编程——TCP的特性之自动重传/流量控制/拥塞控制,一篇说清楚

2024-04-20 05:12

本文主要是介绍网络编程——TCP的特性之自动重传/流量控制/拥塞控制,一篇说清楚,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 1. ARQ自动重传协议
      • 1.1 停止等待ARQ
      • 1.2 连续ARQ
      • 1.3 总结
    • 2. TCP的流量控制
    • 3. TCP的拥塞控制
      • 3.1 慢开始算法
      • 3.2 拥塞避免算法
      • 3.3 快重传算法
      • 3.4 快恢复算法

1. ARQ自动重传协议

自动重传请求(Automatic Repeat-reQuest),通过使用确认和超时这两个机制,在不可靠服务的基础上实现可靠的信息传输,其中包括停止等待ARQ协议和连续ARQ协议

1.1 停止等待ARQ

发送窗口大小为1,接收窗口大小也为1
发送方每发送一个数据包,就要等待接收方返回ack包,如果在定时时间内没收到ack包,则需要重新发送,而这个超时时间,是需要经过RTT往返时延(从发送方发送数据开始,到发送方接收到来自接收方的确认消息)来计算出来的。

当数据正常传输时,发送完M1数据包后,等待接收方的M1数据包ACK,收到后继续发送M2数据包并等待ACK
image.png

当数据传输出错时,假设数据包M1丢失,则一定超时时间后,进行一次重传。
image.png

1.2 连续ARQ

连续ARQ发送方可连续发送多个分组的数据,而不需要像停止等待ARQ一样,等到应答再进行发送,大大提升带宽的利用率。
image.png

1)滑动窗口概念
滑动窗口协议在发送方和接收方之间各自维持一个滑动窗口,两个窗口大小不一定相同。
主要提供TCP的可靠性(基于确认重传机制)以及TCP的流控特性(控制网络传输时的流量,避免拥塞发生)。

滑动窗口其实类似一个收费站,收费站也就是窗口的位置是不变的,数据不停的在进行滑动。
image.png

2)滑动窗口重发机制
发送端维护一个窗口,窗口内有多个分组,分组个数等于窗口的大小,窗口内的分组可以直接连续发送数据,不需要等待接收端返回的ACK,这样可以提升对信道的利用率。

TCP协议提供两种滑动窗口协议:回退(Go-Back-N)以及选择重传(Selective Repeat来解决连续ARQ模式下传输数据出错的问题。
1)回退(Go-Back-N)协议
发送窗口大小 n > 1,接收窗口 = 1,当发生数据丢失时,会重传所有大于最后一个ACK的包
image.png
在这种模式下,发送端会维护一块发送端的数据缓存,当需要重发窗口中的分组报文,便会从缓存里读取数据发送。
这里采用的是累计确认的形式,不像停止等待ARQ,现在不需要对数据帧进行逐个的确认,而是对按序到达的最后一个分组进行确认,假设发送方发送了5个包,但是第三个包丢失,则接收方只返回前两个包的ACK,此时发送方因为不知道后面的三个包有没有发送成功,只能选择这三个包进行重传。
2)选择重传(Selective Repeat)协议
发送窗口大小 > 1,接收窗口大小 > 1,当发生数据丢失,只重传丢失的数据包。
此时已经无法依赖ACK包去做选择重传了,因为ACK只能表示收到了哪些包,但中间的包丢失时,则无法表示。
所以在这里引入了SACK**(Selective Acknowledgement)**,存储在TCP头部的可变选项中,记录接收窗口缓存中还未收到的数据包信息。
image.png
具体例子如下:
image.png

1.3 总结

协议窗口大小是否有序接收具体做法
停止等待ARQ发送=1, 接收=1有序发送窗口每次只能发送一个数据包,然后就停止等待ack包。接收窗口有序的接收数据包,接收成功后发送ack包给发送窗口,如果收到的数据包是无序的,就直接丢弃
连续ARQ-回退协议发送 = N,接收 = 1有序发送窗口每次最多一次性发送n个数据包,接收窗口有序的接收数据包,当接收到有序的数据包后,发送ack包给发送窗口,如果收到的数据包时无序的,就直接丢弃。当数据包丢失的时候,会将发送窗口中的后面的所有数据包都重新发送
连续ARQ-选择重传协议发送 = N,接收 = N无序发送窗口每次最多一次性发送n个数据包,接收窗口无序的接收数据包,当接收到数据包后,发送ack包给发送窗口,ack中会携带SACK信息,也就是接收窗口中的缓存信息。发送端会根据SACK信息来只重传丢失的数据包

2. TCP的流量控制

目的:防止分组丢失进而触发自动重传机制,造成网络流量的浪费。

原理:如果发送者发送数据过快,接收者来不及接收,那么就会有数据分组丢失。为了避免这种分组丢失,接收端会通知发送端它的接收窗口大小(TCP首部中有一个窗口大小值),此时发送者也将发送窗口大小更改为这个值,让接收者来得及接收。

具体案例:假设一开始主机A和B的窗口都是400,那么在发送时,主机A会连续发送400个字节的数据,如果201~300号意外丢失了,主机B会返回响应ACK=1,ack=201,rwnd=300,这表示主机B已接收到201号之前的数据,并将自己的接收窗口设置为300。此时A收到后将发送窗口大小也设置为300,达到流量控制的目的。

3. TCP的拥塞控制

目的:防止过多的数据注入到网络中,避免出现网络负载过大的情况,常用的算法就是:
慢开始、拥塞避免、快重传、快恢复

原理:发送方维持一个拥塞窗口cwnd(congestion window)的状态变量,拥塞窗口的大小取决于网络的拥塞程度,并且动态地在变化。另外考虑到接受方的接收能力 发送方的发送窗口小于或等于拥塞窗口

3.1 慢开始算法

原理:一开始不发送大量的数据,防止网络负载过大,由小到大逐渐增加拥塞窗口的大小,来探测网络的拥塞程度

具体案例:发送方每次经过一个传输轮次之后,拥塞窗口cwnd就直接加倍,这样比直接一下把许多报文注入网络要慢的多。
image.png

3.2 拥塞避免算法

原理:当拥塞窗口cwnd到达慢开始门限ssthresh后,让拥塞窗口缓慢增长,每经过一个传输轮次后,将cwnd值加一,而不是直接加倍。使cwnd已经到达一定值的情况下,网络不容易出现阻塞。

慢开始门限ssthresh与拥塞窗口cwnd的关系
当cwnd<ssthresh时,使用慢开始算法
当cwnd>ssthresh时,改用拥塞避免算法
当cwnd=ssthresh时,慢开始与拥塞避免算法任意

具体案例:拥塞窗口cwnd初始值为1,慢开始门限ssthresh初始值是16
1)在cwnd<ssthresh,执行慢开始算法,cwnd的值在经过一个轮次传输后值翻倍
2)当cwnd>ssthresh,执行拥塞避免算法,每一个轮次,cwnd的值只加1
3)当cmnd = 24时,假设发生网络阻塞,将进行”乘法减小“,ssthresh = cwnd / 2,cwnd为0,重新开始慢开始算法
image.png

3.3 快重传算法

原理:快速进行重传,当接收方在收到一个失序的报文段后就立即发出重复确认,发送方只要一连收到三个重复确认就认为是网络阻塞,立即重传对方尚未收到的报文段,而不必继续等待设置的重传计时器时间到期。(提高网络吞吐量百分之20左右)

具体案例
image.png

3.4 快恢复算法

原理:快速恢复传输,和快重传搭配使用,当发生快重传时,进行”乘法减小“算法,此时不会再进行慢开始算法,而是执行快恢复算法,直接将cwnd设置为ssthresh减半后的值。

具体案例:cwnd为24时,收到三个重复确认,则进行快重传,此时执行”乘法减小“算法,ssthresh = cwnd/2,cwnd = ssthresh,直接便开始了拥塞避免算法。

注意:在TCP Reno版本采用快恢复算法时,慢开始算法只是在TCP连接建立时和网络出现超时时才使用,其他例如收到三个重复的确认,则会执行快恢复算法。
image.png

这篇关于网络编程——TCP的特性之自动重传/流量控制/拥塞控制,一篇说清楚的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/919354

相关文章

SpringBoot请求参数接收控制指南分享

《SpringBoot请求参数接收控制指南分享》:本文主要介绍SpringBoot请求参数接收控制指南,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring Boot 请求参数接收控制指南1. 概述2. 有注解时参数接收方式对比3. 无注解时接收参数默认位置

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构

Android实现两台手机屏幕共享和远程控制功能

《Android实现两台手机屏幕共享和远程控制功能》在远程协助、在线教学、技术支持等多种场景下,实时获得另一部移动设备的屏幕画面,并对其进行操作,具有极高的应用价值,本项目旨在实现两台Android手... 目录一、项目概述二、相关知识2.1 MediaProjection API2.2 Socket 网络

IDEA自动生成注释模板的配置教程

《IDEA自动生成注释模板的配置教程》本文介绍了如何在IntelliJIDEA中配置类和方法的注释模板,包括自动生成项目名称、包名、日期和时间等内容,以及如何定制参数和返回值的注释格式,需要的朋友可以... 目录项目场景配置方法类注释模板定义类开头的注释步骤类注释效果方法注释模板定义方法开头的注释步骤方法注

pytorch自动求梯度autograd的实现

《pytorch自动求梯度autograd的实现》autograd是一个自动微分引擎,它可以自动计算张量的梯度,本文主要介绍了pytorch自动求梯度autograd的实现,具有一定的参考价值,感兴趣... autograd是pytorch构建神经网络的核心。在 PyTorch 中,结合以下代码例子,当你

Python如何自动生成环境依赖包requirements

《Python如何自动生成环境依赖包requirements》:本文主要介绍Python如何自动生成环境依赖包requirements问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录生成当前 python 环境 安装的所有依赖包1、命令2、常见问题只生成当前 项目 的所有依赖包1、

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

Spring Boot项目中结合MyBatis实现MySQL的自动主从切换功能

《SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能》:本文主要介绍SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能,本文分步骤给大家介绍的... 目录原理解析1. mysql主从复制(Master-Slave Replication)2. 读写分离3.

微信公众号脚本-获取热搜自动新建草稿并发布文章

《微信公众号脚本-获取热搜自动新建草稿并发布文章》本来想写一个自动化发布微信公众号的小绿书的脚本,但是微信公众号官网没有小绿书的接口,那就写一个获取热搜微信普通文章的脚本吧,:本文主要介绍微信公众... 目录介绍思路前期准备环境要求获取接口token获取热搜获取热搜数据下载热搜图片给图片加上标题文字上传图片

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子