【Linux网络编程】传输协议UDP

embedded/2024/12/29 15:04:18/

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站


在这里插入图片描述

🌈个人主页: 南桥几晴秋
🌈C++专栏: 南桥谈C++
🌈C语言专栏: C语言学习系列
🌈Linux学习专栏: 南桥谈Linux
🌈数据结构学习专栏: 数据结构杂谈
🌈数据库学习专栏: 南桥谈MySQL
🌈Qt学习专栏: 南桥谈Qt
🌈菜鸡代码练习: 练习随想记录
🌈git学习: 南桥谈Git

🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈
本科在读菜鸡一枚,指出问题及时改正


文章目录

  • 再谈端口号
    • 端口号的范围划分
    • 知名端口号
    • 一个进程是否能绑定多个端口号?
    • 一个端口号是否可以被多个进程绑定?
    • 理解端口号和进程的关系
  • UDP协议
    • UDP协议格式
    • UDP特点
    • 面向数据报
    • UDP缓冲区
    • 使用注意事项
    • 基于UDP的应用层协议
    • 进一步理解UDP报头
    • 进一步理解报文

再谈端口号

端口号标识了一个主机上进行通信的不同的应用程序

TCP/IP 协议中,使用 五元组 (5-tuple) 来唯一标识一条网络通信。这个五元组包含以下五个信息元素:

  1. 源 IP 地址:发送方设备的 IP 地址,用于标识通信的源。
  2. 源端口号:发送方应用程序的端口号,标识发送方在其设备上的具体应用进程。
  3. 目的 IP 地址:接收方设备的 IP 地址,用于标识通信的目标。
  4. 目的端口号:接收方应用程序的端口号,标识接收方设备上的具体应用进程。
  5. 协议号:用于标识协议类型的字段,通常用于区分不同的传输层协议。例如,TCP 使用协议号 6,UDP 使用协议号 17。

可以通过netstat -nltp查看:
在这里插入图片描述

在这里插入图片描述

端口号的范围划分

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

知名端口号

在Linux系统中,可通过vim /etc/services来查看知名端口号:

在这里插入图片描述

一个进程是否能绑定多个端口号?

可以。以TCP为例,可以创建多个listen套接字,用的是不同的端口号。一个服务器可以创建两个端口号,一个进行发送数据,另一个进行发送控制命令。

一个端口号是否可以被多个进程绑定?

原则上不可以。需要保证端口号与服务之间的唯一性。

理解端口号和进程的关系

进程在Linux内核中实际上是一个struct task_struct,这就是描述进程的一个结构体。操作系统内部维护了一张哈希表,哈希表对应的key对应端口号,value对应进程PCB的地址。在进行bind绑定的时候是将进程PCB地址与哈希表的key端口号进程绑定,换言之,所谓的绑定就是将PCB地址和端口号构建在哈希表中。底层收到数据,读取到目的端口号就可以找到对应的进程,就可以将数据交给这个进程。因此一个端口号只能被一个进程绑定,需要保持key值唯一。

UDP协议

UDP协议格式

UDP报头一定是一个结构体

在这里插入图片描述

16 位 UDP 长度, 表示整个数据报(UDP 首部+UDP 数据)的最大长度

UDP特点

  • 无连接: 知道对端的 IP 和端口号就直接进行传输, 不需要建立连接;
  • 不可靠: 没有确认机制, 没有重传机制; 如果因为网络故障该段无法发到对方,UDP 协议层也不会给应用层返回任何错误信息;
  • 面向数据报: 不能够灵活的控制读写数据的次数和数量;

面向数据报

应用层交给 UDP 多长的报文, UDP 原样发送, 既不会拆分, 也不会合并

UDP缓冲区

  • UDP 没有真正意义上的 发送缓冲区. 调用 sendto 会直接交给内核, 由内核将数据传给网络层协议进行后续的传输动作;
  • UDP 具有接收缓冲区. 但是这个接收缓冲区不能保证收到的 UDP 报的顺序和发送 UDP 报的顺序一致; 如果缓冲区满了, 再到达的 UDP 数据就会被丢弃
    在这里插入图片描述

UDP不需要可靠性保证,不需要丢包重传,只需要添加报头,UDP的报头很简单只有8个字节,添加报头后直接发送,因此不需要放在发送缓冲区保存起来。

虽然UDP不需要保证可靠性,但是起码需要保证报文不会大面积丢失,因此提供一个接收缓冲区,当上层正在读取UDP报文,操作系统可以继续接收UDP数据,这样也会在一定程度提高效率。接收缓冲区一旦写满了,这样再接收到的数据就会丢失。

UDP 的 socket 既能读, 也能写, 这个概念叫做 全双工

使用注意事项

UDP 协议首部中有一个 16 位的最大长度. 也就是说一个 UDP 能传输的数据最大长度是 64K(包含 UDP 首部)。如果我们需要传输的数据超过 64K, 就需要在应用层手动的分包, 多次发送, 并在接收端手动拼装。

基于UDP的应用层协议

  • NFS: 网络文件系统
  • TFTP: 简单文件传输协议
  • DHCP: 动态主机配置协议
  • BOOTP: 启动协议(用于无盘设备启动)
  • DNS: 域名解析协议

进一步理解UDP报头

UDP报头实际上也是一个结构体,它的具体内容如下:

struct udphdr {__be16 source;      // 源端口号__be16 dest;        // 目标端口号__be16 len;         // UDP 数据长度(包括头部和数据部分)__be16 check;       // 校验和(UDP 校验和)
};

当我们需要拿到对应成员的数据时,使用二进制格式进行序列化即可,struct udphdr* h指向对应的成员。

进一步理解报文

在接收方或者发送方,通信双方的操作系统内会同时存在很多报文,要么向上交付,要么向下交付,UDP报文会接受很多数据。在进行通信时,需要对报文进行管理,先描述,再组织

描述报文的结构体为struct sk_buff,内部有数据包的头信息、数据、缓冲区等。将应用层数据拷贝到缓冲区实际上是将应用层数据拷贝到缓冲区,此时有了数据。

struct sk_buff {struct sk_buff *next;            // 指向下一个 sk_buff(链表结构)struct sk_buff *prev;            // 指向上一个 sk_buff(链表结构)struct net_device *dev;          // 网络设备(接口),即接收数据包的网卡unsigned int len;                // 数据包的长度unsigned int data_len;           // 有效数据的长度(不包括协议头)unsigned char *data;             // 指向数据区的指针(即数据包的负载部分)unsigned char *head;             // 数据包的开始地址(头部)unsigned char *tail;             // 数据包的尾部(通常指向空闲区域的结束)unsigned char *end;              // 数据包的结束地址(缓冲区末尾)unsigned int truesize;           // sk_buff 的实际大小(包括头部、数据区和尾部)struct sk_buff *next_free;       // 用于内核的 sk_buff 内存池中的链表struct sock *sk;                // 套接字结构体(用于连接和数据传输)unsigned int protocol;           // 数据包的协议类型(如 IPv4、IPv6、ARP 等)__be16 transport_header;        // 运输层协议头部位置(如 UDP、TCP)__be16 network_header;          // 网络层协议头部位置(如 IP)__be16 mac_header;              // 链路层协议头部位置(如以太网)// 更多字段和标志用于特定功能(例如 QoS、优先级、标记等)
};

然后添加UDP报头:

struct sk_buff bufffer;
(struct udphdr*) buffer->head-=sizeof(struct udphdr);
buffer->head->source=12345;
buffer->head->dest=8888;
buffer->head->len=100;
buffer->head->check=Check();

在这里插入图片描述

后续需要再添加报文时,head就继续往前移动即可。


在这里插入图片描述


http://www.ppmy.cn/embedded/145562.html

相关文章

【笔记】分布式任务调度平台XXL-JOB

这篇笔记主要记录以下内容: (1)第一次启动xxl-job的过程 (2)模块、文件、数据库(表和字段)的作用 (3)极少的源码解读(XxlJobConfig) 有点像实…

CentOS7环境安装php

直接安装 yum -y install php CentOS7默认安装是php5,现在php已有8.3版本 先查看php -v 版本 如果是低版本,可以删除 yum remove php yum remove php-fpm yum remove php-common 一、添加REMI存储库 yum install epel-release yum install -y …

游戏引擎学习第43天

仓库 https://gitee.com/mrxiao_com/2d_game 介绍运动方程 今天我们将更进一步,探索运动方程,了解真实世界中的物理,并调整它们,以创建一种让玩家感觉愉悦的控制体验。这并不是在做一个完美的物理模拟,而是找到最有趣…

0002.基于springboot +layui二手物品交易平台

适合初学同学练手项目,部署简单,代码简洁清晰; 注:当前项目架构使用前后端未分离哦! 一、系统架构 前端:layui| html 后端:springboot | mybatis-plus 环境:jdk1.8 | mysql | maven 二、代…

计算机网络技术基础:1.计算机网络的产生与发展

从1946年世界上第一台计算机ENIAC的诞生,计算机网络的发展大体可分为以下4个阶段。 一、第一代计算机网络——面向终端的计算机网络 第一代计算机网络也称面向终端的计算机网络,它是以主机为中心的通信系统。这样的系统中,除一台中心计算机&…

深入解析Vue3响应式系统:从Proxy实现到依赖收集的核心原理

深入解析Vue3响应式系统:从Proxy实现到依赖收集的核心原理 响应式系统的基本原理 作为一个热门的JavaScript框架,Vue在3.x版本中引入了基于Proxy的响应式系统。这个系统的核心思想是利用Proxy对象拦截对数据的访问和修改,从而实现数据的自动更…

中电金信携手中远海科,共启贸易金融数智新篇章

在数智化转型成为驱动经济社会高质量发展的新引擎背景下,“数智方案”栏目聚焦金融等国计民生重点行业场景,依托中电金信“源启筑基咨询引领应用重构”的产品及服务体系,输出市场洞察和行业解决方案、应用案例,旨在全面推动行业IT…

opencv下载安装及VS配置(笔记)

1、opencv下载及安装 官网地址:https://opencv.org/,点击Releases进入下载界面: 根据自己的需要下载相应的版本,这里我下载的是opencv-4.10版本: 找到下载的exe文件: 双击安装,选择安装路径&…