部署DeepSeek模型,进群交流最in玩法!
立即加群
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >【计网】从零开始理解UDP协议 --- 理解端口号和UDP结构

【计网】从零开始理解UDP协议 --- 理解端口号和UDP结构

作者头像
叫我龙翔
发布于 2024-10-16 08:07:17
发布于 2024-10-16 08:07:17
6060
举报

从零开始理解UDP协议

1 再谈端口号

之前我们讲过服务器上的端口号和服务器的进程是绑定的!客户端的进程与客户端的端口号也是绑定的!再通过IP地址,就可以快速找到网络中需要进行通信的进程!

用 “源 IP”, “源端口号”, “目的 IP”, “目的端口号”, “协议号” 这样一个五元组来标识一个通信,可以明确目标进程和来源进程以及通信协议

端口号范围划分 • 0 - 1023:知名端口号, HTTP,FTP,SSH 等这些广为使用的应用层协议, 他们的端口号都是固定的。 • 1024 - 65535:操作系统动态分配的端口号。客户端程序的端口号, 就是由操作系统从这个范围分配的。

我们之前测试的时候都是绑定的8888端口,如果今天绑定0 - 1023的端口,就会绑定失败:

只有我们使用超级用户的权限,我们才可以绑定0-1023端口号!普通用户是不能随便绑定知名端口号的!知名端口号都是与特定的服务联系在一起的!也就是说未来客户端可以在不知道端口号的情况下链接特定服务时直接使用知名端口号!

有些服务器是非常常用的, 为了使用方便, 人们约定一些常用的服务器, 都是用以下这些固定的端口号: • ssh 服务器, 使用 22 端口 • ftp 服务器, 使用 21 端口 • telnet 服务器, 使用 23 端口 • http 服务器, 使用 80 端口 • https 服务器, 使用 443

现在有两个问题

  1. 一个进程是否可以bind多个端口号? 可以的!我们要的是端口号到服务的唯一性,一个进程可以创建多个Socket,每个Socket都可以绑定一个端口号!
    • 多线程/多进程: 进程可以创建多个线程或子进程,每个线程或子进程可以有自己的socket,并且每个socket可以绑定到不同的端口号。
    • 多路复用: 进程可以使用I/O多路复用技术(如select, poll, epoll等),在单个进程中同时处理多个socket。尽管这些socket可能监听不同的端口,但它们是由操作系统统一管理的,而不是进程“绑定”了多个端口。
  2. 一个端口号是否可以被多个进程bind? 原则上是不可以的!因为服务和对应端口是紧密联系的!除非使用了特殊的套接字选项。

理解端口号和进程的关系: 在操作系统内部有这样一个描述进程的结构体task_struct !操作系统还有这样一个哈希表,K为端口号,V为进程的结构体task_struct !进绑定时就是将端口号与进程结构体建立哈希关系!这样传输层从网络层解析出来端口号时,你可以找到对应的进程,进入应用层!这样也就理解了一个端口号不能被多个进程bind,不然就产生哈希冲突了!

2 理解UDP 报头结构

协议是一种约定,是双方都认识的结构化数据!在学习应用层时,我们自己设计了自己的结构体作为协议!那么UDP也就是一种结构体!

  1. 任何协议都要解决如何经报头与有效载荷进行分离:UDP这里报头是固定的前8个字节!可以开始将报头与有效载荷进行分离!
  2. 如何将有效载荷进行分用!根据UDP报头中的16位端口号就可以找到对应的进程,然后进行分用!

我们来看源代码中的UDP报头结构:

这个结构体十分的简单奥!其中的UDP校验和是用来检验数据是否正常的,如果数据校验出错,就会直接丢弃数据!所以说UDP协议是不可靠的

作为一种协议,那么是不是就要进行序列化与反序列化?那为什么没有看到UDP的序列化和反序列化?其实报头就是一个结构体变量,直接加到报文前,读取是直接进行二进制读取获取到结构体变量!**但是应用层是不能这样写的!应用层涉及不同语言,大小端机器等很多问题!!!**而内核没有业务!报文该是多少就是多少,不会增添新的内容!并且双方操作系统都是C语言写的!都要先转大端序列!那么内核就能直接读取结构体变量,并且不会出现问题了!!!

3 UDP 的特点

UDP 传输的过程类似于寄信

  • 无连接:知道目的端的 IP 和端口号就直接进行传输,不需要建立连接!
  • 不可靠:没有确认机制, 没有重传机制;如果因为网络故障该段无法发到对方,UDP 协议层也不会给应用层返回任何错误信息!
  • 面向数据报: 不能够灵活的控制读写数据的次数和数量!接收到报文只需要进行序列化与反序列化!不需要判断是否读取到完整信息! 应用层交给 UDP 多长的报文,UDP 原样发送,既不会拆分,也不会合并。用 UDP 传输 100 个字节的数据:如果发送端调用一次 sendto,发送 100 个字节,那么接收端也必须调用对应的一次 recvfrom,接收 100 个字节。而不能循环调用 10 次 recvfrom,,每次接收 10 个字节;

4 UDP 的缓冲区

UDP 没有真正意义上的发送缓冲区。调用 sendto 会直接交给内核,由内核将数据传给网络层协议进行后续的传输动作。应用层直接就通过内核发送!因为UDP没有重传机制,发送只会进行一次,并且UDP的报头结构很简单,应用层的报文直接加上8字节即可。那么就不需要同一个缓冲区来进行管理了!

但UDP 具有接收缓冲区。UDP的接收缓冲区可以提高效率,执行任务时依旧可以读取数据!但是这个接收缓冲区不能保证收到的 UDP 报的顺序和发送 UDP 报的顺序一致。如果缓冲区满了,再到达的 UDP 数据就会被丢弃!

UDP的发送与接收是独立的,那么自然就支持全双工通信了!

在网络通信过程中,操作系统会不断的接收报文,应用层产生报文。所以OS中可能同时存在大量的报文,这些报文可能正在被向上交付,也可能被向下交付!所以操作系统就要对报文进行管理!一个完整的报文不仅仅是报头和有效载荷,还需要一个管理报文的结构 !

管理报文的结构化字段struct_sk_buff内部一个指针指向下一个报文。是通过链式结构(Linux系统是双链表)的增删查改进行管理的! struct_sk_buff内部有指向数据的指针。当报文向下传输时,会先将报文内部的数据写到下一层的一个缓冲区中,注意是写到缓冲区的中间位置。然后将head指针向前移动相应报头大小,之后就可以在head这片空间内写入新的报头了!向上传输就是将head指针向后移动除去报头即可!

5 UDP 使用注意事项

我们注意到,UDP 协议首部中有一个 16 位的最大长度。 也就是说一个 UDP 能传输的数据最大长度是 64K(包含 UDP 首部)。单个报文的长度不能超过64K!

然而 64K 在当今的互联网环境下,是一个非常小的数字

如果我们需要传输的数据超过 64K, 就需要在应用层手动的分包,多次发送, 并在接收端手动拼装。多次发送,接收端手动拼装!

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-10-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
传输层协议UDP详解
前面已经讲过,HTTP协议是应用层协议,在此之前,我们短暂的认为HTTP是直接通过应用层与外界通信的。但是我们要知道,应用层需要向下将数据传到传输层,再由传输层向下传送。最终才能通过网络传输到接收方。
小灵蛇
2024/10/17
2990
传输层协议UDP详解
【网络编程】十、详解 UDP 协议
​ 在学习 HTTP 等应用层协议时,为了便于理解,可以简单的认为 HTTP 协议是将请求和响应直接发送到了网络当中。但实际应用层需要先将数据交给传输层,再由传输层对数据做进一步处理后再将数据继续向下进行交付,该过程贯穿整个网络协议栈,最终才能将数据发送到网络当中。
利刃大大
2025/05/17
1350
【网络编程】十、详解 UDP 协议
传输层:UDP协议
端口号标识的是一个主机上进行通信的不同的应用程序,通过IP+PORT,便能够确认全网唯一一个进程。
二肥是只大懒蓝猫
2023/10/13
3910
传输层:UDP协议
传输层协议——UDP
ssh服务器, 使用22端口 ftp服务器, 使用21端口 telnet服务器,使用23端口 http服务器, 使用80端口 https服务器, 使用443端口
lovevivi
2023/11/27
2780
传输层协议——UDP
浅谈面向数据报的协议-UDP协议
协议用通俗的话来说就是约定,因为计算机之间的传输媒介是光信号和电信号. 通过 "频率" 和 "强弱" 来表示 0 和 1 这样的信息. 要想传递各种不同的信息, 就需要约定好双方的数据格式.
用户10923087
2024/08/09
2480
浅谈面向数据报的协议-UDP协议
【Linux网络编程】传输协议UDP
TCP/IP 协议中,使用 五元组 (5-tuple) 来唯一标识一条网络通信。这个五元组包含以下五个信息元素:
南桥
2024/12/14
4270
【Linux网络编程】传输协议UDP
【Linux】传输层协议:UDP和TCP
1. 在网络通信中,通信的本质实际就是两台主机上的进程在网络环境中进行通信,也就是数据的传输,而我们总说TCP/IP协议栈,这两个协议分别解决了两个重要的问题,即一台主机如何在网络环境中标定自己的唯一性,一台主机中的某个进程如何在主机内部标定自己的唯一性,实际就是通过网络层协议IP地址和传输层协议端口号port来解决这两个问题的。
举杯邀明月
2023/10/17
1.4K0
【Linux】传输层协议:UDP和TCP
【Linux】深入理解传输层:端口号、UDP协议及其应用场景
端口号(Port)标识了一个主机上进行通信的不同的应用程序。
用户11316056
2024/11/26
5040
【Linux】深入理解传输层:端口号、UDP协议及其应用场景
【Linux网络】网络基础:传输层UDP/TCP协议(一)
前言:传输层协议,特别是用户数据报协议(UDP)和传输控制协议(TCP),是网络通信中最为基础也最为重要的部分。它们不仅决定了数据的传输方式,还影响着数据的可靠性、顺序性和实时性。对于想要深入了解互联网运行机制、掌握网络通信技术的朋友们来说,学习UDP/TCP协议无疑是必经之路。
Eternity._
2024/12/13
1750
【Linux网络】网络基础:传输层UDP/TCP协议(一)
【计算机网络】UDP/TCP 协议
端口号(Port)标识了一个主机上进行通信的不同的应用程序。在 TCP/IP 协议中, 用 “源IP”, “源端口号”, “目的IP”, “目的端口号”, “协议号” 这样一个五元组来标识一个通信(可以通过netstat -n查看)。
YoungMLet
2024/03/16
2570
【计算机网络】UDP/TCP 协议
【Linux】:传输层协议 UDP
🔥 之前在这篇文章 初识网络 中说过关于传输层的内容,以及在 Socket编程应用层UDP 也做过关于 UDP 的练习,如下:
IsLand1314
2025/02/04
5200
【Linux】:传输层协议 UDP
【计网】从零开始掌握序列化 --- 实现网络计算器项目
最重要的是将Socket进行了程序重构,具体的细节在TCP协议中讲解过。这样将通信功能彻底解耦出来:
叫我龙翔
2024/09/25
970
【计网】从零开始掌握序列化 --- 实现网络计算器项目
Linux网络-UDP/TCP协议详解
注:端口号大部分都是16位的,其根本原因就是因为传输层协议当中的端口号就是16位的
用户9645905
2022/11/15
1.8K0
Linux网络-UDP/TCP协议详解
【Linux】TCP网络套接字编程+协议定制+序列化和反序列化
1. 为了让我们的代码更规范化,所以搞出了日志等级分类,常见的日志输出等级有DEBUG NORMAL WARNING ERROR FATAL等,再配合上程序运行的时间,输出的内容等,公司中就是使用日志分类的方式来记录程序的输出,方便程序员找bug。 实际上在系统目录/var/log/messages文件中也记录了Linux系统自己的日志输出,可以看到我的Linux系统中之前在使用时产生了很多的error和warning,我们的代码也可以搞出来这样的输出日志信息到文件的功能。
举杯邀明月
2023/10/17
4660
【Linux】TCP网络套接字编程+协议定制+序列化和反序列化
【Linux】: 传输层协议 TCP
🔥 之前在这篇文章 传输层协议 UDP 中已经说过关于传输层的部分内容,现在我们来了解一下传输层 TCP 的内容吧
IsLand1314
2025/02/20
4130
【Linux】: 传输层协议 TCP
Linux网络UDP与TCP
端口号(Port)标识了一个主机上进行通信的不同的应用程序; 在 TCP/IP 协议中, 用 “源 IP”, “源端口号”, “目的 IP”, “目的端口号”, “协议号” 这样一个五元组来标识一个通信(可以通过 netstat -n 查看);
有礼貌的灰绅士
2025/04/21
2280
Linux网络UDP与TCP
传输层协议TCP详解(上篇)
前面我们已经讲过为什么read、write、recv、send 和 tcp 支持全双工,这其中涉及到发送缓冲区和接收缓冲区(点此查看)。我们创建的TCP套接字,实际上sockfd会指向一个操作系统给分配好的socket file control block(socket文件控制块),而这个socket文件控制块就是维护发送缓冲区和接收缓冲区的。
小灵蛇
2024/10/29
9490
传输层协议TCP详解(上篇)
TCP协议和UDP协议
1.1.2每一条TCP连接只能有两个端点,每一条TCP链接只能是点对点的(一对一)
全栈程序员站长
2022/06/26
1.4K0
TCP协议和UDP协议
初识Linux · 传输层协议UDP
前文我们介绍了UDP的代码使用,流程也是非常简单的,创建了socket,然后填充对应的信息,最后bind,双方就可以进行通信了。通过代码的编写我们发现UDP是简单的,那么我们进入到源码方面学习UDP也是较为简单的。
_lazy
2025/05/08
1360
初识Linux · 传输层协议UDP
【计网】深入理解网络通信:端口号、Socket编程及编程接口
数据传输到主机是目的吗?不是的。因为数据是给人用的。比如:聊天是人在聊天,下载是人在下载,浏览网页是人在浏览!但是人是怎么看到聊天信息的呢?怎么执行下载任务呢?怎么浏览网页信息呢?通过启动的 qq,迅雷,浏览器。 而启动的 qq,迅雷,浏览器都是进程。换句话说,进程是人在系统中的代表,只要把数据给进程,人就相当于就拿到了数据。
用户11316056
2024/11/19
3630
【计网】深入理解网络通信:端口号、Socket编程及编程接口
相关推荐
传输层协议UDP详解
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档