UDP数据包大小问题

news/2024/5/18 15:14:12 标签: UDP, 数据包

在进行UDP编程的时候,我们最容易想到的问题就是,一次发送多少bytes好?

       当然,这个没有唯一答案,相对于不同的系统,不同的要求,其得到的答案是不一样的,这里仅对像ICQ一类的发送聊天消息的情况作分析,对于其他情况,或许也能得到一点帮助: 
首先,我们知道,TCP/IP通常被认为是一个四层协议系统,包括链路层,网络层,传输层,应用层.UDP属于运输层,下面我们由下至上一步一步来看: 
以太网(Ethernet)数据帧的长度必须在46-1500字节之间,这是由以太网的物理特性决定的.这个1500字节被称为链路层的MTU(最大传输单元).但这并不是指链路层的长度被限制在1500字节,其实这个MTU指的是链路层的数据区.并不包括链路层的首部和尾部的18个字节.所以,事实上,这个1500字节就是网络层IP数据报的长度限制.因为IP数据报的首部为20字节,所以IP数据报的数据区长度最大为1480字节.而这个1480字节就是用来放TCP传来的TCP报文段或UDP传来的UDP数据报的.又因为UDP数据报的首部8字节,所以UDP数据报的数据区最大长度为1472字节.这个1472字节就是我们可以使用的字节数。:) 
当我们发送的UDP数据大于1472的时候会怎样呢?这也就是说IP数据报大于1500字节,大于 MTU.这个时候发送方IP层就需要分片(fragmentation).把数据报分成若干片,使每一片都小于MTU.而接收方IP层则需要进行数据报的重组.这样就会多做许多事情,而更严重的是,由于UDP的特性,当某一片数据传送中丢失时,接收方便无法重组数据报.将导致丢弃整个UDP数据报。 
因此,在普通的局域网环境下,我建议将UDP的数据控制在1472字节以下为好. 
进行Internet编程时则不同,因为Internet上的路由器可能会将MTU设为不同的值.如果我们假定MTU为1500来发送数据的,而途经的某个网络的MTU值小于1500字节,那么系统将会使用一系列的机制来调整MTU值,使数据报能够顺利到达目的地,这样就会做许多不必要的操作.鉴于 Internet上的标准MTU值为576字节,所以我建议在进行Internet的UDP编程时.最好将UDP的数据长度控件在548字节 (576-8-20)以内.

理论上,IP数据报的最大长度是65535字节,这是由IP首部16比特总长度字段所限制的。去除20字节的IP首部和8个字节的UDP首部,UDP数据报中用户数据的最长长度为65507字节。但是,大多数实现所提供的长度比这个最大值小。 
我们将遇到两个限制因素。

第一,应用程序可能会受到其程序接口的限制。socket API提供了一个可供应用程序调用的函数,以设置接收和发送缓存的长度。对于UDP socket,这个长度与应用程序可以读写的最大UDP数据报的长度直接相关。现在的大部分系统都默认提供了可读写大于8192字节的UDP数据报(使用这个默认值是因为8192是NFS读写用户数据数的默认值)。 
第二个限制来自于TCP/IP的内核实现。可能存在一些实现特性(或差错),使IP数据报长度小于65535字节。 
在SunOS 4.1.3下使用环回接口的最大IP数据报长度是32767字节。比它大的值都会发生差错。 
但是从BSD/386到SunOS 4.1.3的情况下,Sun所能接收到最大IP数据报长度为32786字节(即32758字节用户数据)。 
在Solaris 2.2下使用环回接口,最大可收发IP数据报长度为65535字节。 
从Solaris 2.2到AIX 3.2.2,发送的最大IP数据报长度可以是65535字节。很显然,这个限制与源端和目的端的实现有关。 
主机必须能够接收最短为576字节的IP数据报。在许多UDP应用程序的设计中,其应用程序数据被限制成512字节或更小,因此比这个限制值小。 
由于IP能够发送或接收特定长度的数据报并不意味着接收应用程序可以读取该长度的数据。因此,UDP编程接口允许应用程序指定每次返回的最大字节数。如果接收到的数据报长度大于应用程序所能处理的长度,那么会发生什么情况呢?不幸的是,该问题的答案取决于编程接口和实现。 
典型的Berkeley版socket API对数据报进行截断,并丢弃任何多余的数据。应用程序何时能够知道,则与版本有关(4.3BSD Reno及其后的版本可以通知应用程序数据报被截断)。 
SVR4 下的socket API(包括Solaris 2.x) 并不截断数据报。超出部分数据在后面的读取中返回。它也不通知应用程序从单个UDP数据报中多次进行读取操作。TLI API不丢弃数据。相反,它返回一个标志表明可以获得更多的数据,而应用程序后面的读操作将返回数据报的其余部分。在讨论TCP时,我们发现它为应用程序提供连续的字节流,而没有任何信息边界。TCP以应用程序读操作时所要求的长度来传送数据,因此,在这个接口下,不会发生数据丢失。

在进行UDP编程的时候,我们最容易想到的问题就是,一次发送多少bytes好? 
当然,这个没有唯一答案,相对于不同的系统,不同的要求,其得到的答案是不一样的,我这里仅对 
像ICQ一类的发送聊天消息的情况作分析,对于其他情况,你或许也能得到一点帮助: 
首先,我们知道,TCP/IP通常被认为是一个四层协议系统,包括链路层,网络层,运输层,应用层. 
UDP属于运输层,下面我们由下至上一步一步来看: 
以太网(Ethernet)数据帧的长度必须在46-1500字节之间,这是由以太网的物理特性决定的. 
这个1500字节被称为链路层的MTU(最大传输单元). 
但这并不是指链路层的长度被限制在1500字节,其实这这个MTU指的是链路层的数据区. 
并不包括链路层的首部和尾部的18个字节. 
所以,事实上,这个1500字节就是网络层IP数据报的长度限制. 
因为IP数据报的首部为20字节,所以IP数据报的数据区长度最大为1480字节. 
而这个1480字节就是用来放TCP传来的TCP报文段或UDP传来的UDP数据报的. 
又因为UDP数据报的首部8字节,所以UDP数据报的数据区最大长度为1472字节. 
这个1472字节就是我们可以使用的字节数。:)

当我们发送的UDP数据大于1472的时候会怎样呢? 
这也就是说IP数据报大于1500字节,大于MTU.这个时候发送方IP层就需要分片(fragmentation). 
把数据报分成若干片,使每一片都小于MTU.而接收方IP层则需要进行数据报的重组. 
这样就会多做许多事情,而更严重的是,由于UDP的特性,当某一片数据传送中丢失时,接收方便 
无法重组数据报.将导致丢弃整个UDP数据报.

因此,在普通的局域网环境下,我建议将UDP的数据控制在1472字节以下为好. 
进行Internet编程时则不同,因为Internet上的路由器可能会将MTU设为不同的值. 
如果我们假定MTU为1500来发送数据的,而途经的某个网络的MTU值小于1500字节,那么系统将会使用一系列的机 
制来调整MTU值,使数据报能够顺利到达目的地,这样就会做许多不必要的操作.

鉴于Internet上的标准MTU值为576字节,所以我建议在进行Internet的UDP编程时. 
最好将UDP的数据长度控件在
548字节(576-8-20)以内.

---------------------------------------------------------------转载分割线------------------------------------------------------

经测试,局域网环境下,UDP包大小为1024*8,速度达到2M/s,丢包情况理想.

外网环境下,UDP包大小为548,速度理想,丢包情况理想.


http://www.niftyadmin.cn/n/1242281.html

相关文章

Day156.数据访问②-NoSQL -SpringBoot2

数据访问 【2】NoSQL Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库、缓存和消息中间件。 它支持多种类型的数据结构,如 字符串(strings), 散列&#x…

【C语言有什么用?①】从零开始撸一个用户态模拟文件系统

☘写在前面☘ 学习一个语言最好的方法是做一个小项目,这个项目不需要多么复杂,但是一定能激发你的学习兴趣。让我们话不多说,开始吧 本文将带你手撸一个磁盘组织方式的模拟,你将学到 📝 c语言指针的使用 &#x1f4dd…

取消 CDockablePane 自动保存界面配置到注册表

因为MFC默认会把的配置信息保存到注册表,如果每次启动程序还原界面,需要手动关闭保存。 默认保存路径: SetRegistryKey(_T("Local AppWizard-Generated Applications"))不影响程序性能 只有下面有写操作才会写入到这个位置&#xf…

Qt 小例子学习15 - 将文件拖放到QDialog内的 QTreeWidget 中

Qt 小例子学习15 - 将文件拖放到QDialog内的 QTreeWidget 中 FilesTreeWidget.h #ifndef TREEWIDGET_H #define TREEWIDGET_H#include <QDropEvent> #include <QFileInfo> #include <QMimeData> #include <QTreeWidget>#include <QDebug>class…

Day157.单元测试 -SpringBoot2

单元测试 一、JUnit5 的变化 Spring Boot 2.2.0 版本开始引入 JUnit 5 作为单元测试默认库 作为最新版本的JUnit框架&#xff0c;JUnit5与之前版本的Junit框架有很大的不同。由三个不同子项目的几个不同模块组成。 JUnit 5 JUnit Platform JUnit Jupiter JUnit Vintage J…

【C语言有什么用?②】制作一个多线程词频统计工具

☘写在前面☘ 学习一个语言最好的方法是做一个小项目&#xff0c;这个项目不需要多么复杂&#xff0c;但是一定能激发你的学习兴趣。让我们话不多说&#xff0c;开始吧 本文将带你手撸一个多线程词频统计工具&#xff0c;你将学到 &#x1f4dd; 如何创建多线程 &#x1f4dd;…

VS2008中CDockablePane停靠框

VS2008中对于MFC的MDI程序界面做了增强&#xff0c;支持很炫的的布局效果&#xff0c;之前还在说不能用CDialogBar类&#xff0c;现在完全可以使用CDockablePane类替代了&#xff0c;现在只需要新建一个继承自CDockablePane的新类&#xff0c;就可以在其上添加各种控件了&#…

Qt 小例子学习16 - 登陆界面

Qt 小例子学习16 - 登陆界面 Login.h #ifndef LOGIN_H #define LOGIN_H#include <QObject> #include <QDebug> class Login : public QObject {Q_OBJECTQ_PROPERTY(QString username READ username WRITE setUsername NOTIFY usernameChanged)Q_PROPERTY(QString…