Linux 传输层UDP

news2024/10/6 1:43:57

文章目录

  • 一、再谈端口号
    • 查看知名端口号和网站的强关联信息?
    • 一个进程是否可以bind多个端口号?
    • 一个端口号是否可以被多个进程bind?
    • 理解进程和端口的关系?
  • 二、UDP协议
    • 协议格式
      • 16位源端口号:
      • 16位目的端口号:
      • 16位UDP长度:
      • 16位UDP校验和:
      • UDP数据字段:
      • 如何将报头和有效载荷进行分离(封装)?
      • 如何将有效载荷进行分用?
    • UDP的特点
      • 面向数据报:
    • UDP的缓冲区
    • UDP的注意事项
    • UDP深刻理解
      • UDP报头:
      • UDP报文:
  • 三、编写代码
    • UDP.hpp
    • client.cc
    • Server.cc

一、再谈端口号

在传输层和网络层协议中,源IP、目的IP、源端口、目的端口、协议号,用来标识一个通信。

查看知名端口号和网站的强关联信息?

cat /etc/services

一个进程是否可以bind多个端口号?

可以。

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

不可以。

理解进程和端口的关系?

进程存储在一个hash表中,而进程的key值就是端口号。所以一个端口号只能被一个进程bind,保证了key值的唯一性。

image-20241005162401842

二、UDP协议

协议格式

image-20241005162713192

16位源端口号:

表示发送方端口号,占用2个字节。在要求对方回信时选用,不要求时可使用全0。

16位目的端口号:

表示接收方端口号,占用2个字节。在终点交付报文时必须使用。

16位UDP长度:

表示UDP数据报文的长度,包括头部和数据部分,占用2个字节。其最小值是8(仅有首部),最大值为65535字节(包括报文在内的用户数据报的总长度)。由于UDP报头长度固定为8字节,因此实际UDP报文的数据长度最大为65527字节。

16位UDP校验和:

用于检测UDP数据报在传输过程中是否有错误,占用2个字节。如果出错则将报文丢弃。— UDP不保证数据的可靠性。

UDP数据字段:

UDP数据字段占用0个或多个字节,是实际传输的数据部分。

如何将报头和有效载荷进行分离(封装)?

因为UDP协议格式的前8个字节都是固定的,所以只需要将前8个字节进行分离,这样就能得到报文的有效载荷。

如何将有效载荷进行分用?

从前8个字节中,找到目的端口,即可将找到需要启动的PCB进程,然后只需要将有效载荷交给该进程即可。

UDP的特点

  • 无连接。
  • 不可靠。没有确认机制,没有重传机制。
  • 面向数据报。— 发送多少次,就必须接收多次;发送多少字节,就必须一次性接收多少字节。

面向数据报:

应用层交给 UDP 多长的报文, UDP 原样发送, 既不会拆分, 也不会合并;
例如:UDP 传输 100 个字节的数据:如果发送端调用一次 sendto, 发送 100 个字节, 那么接收端也必须调用对应的
次 recvfrom, 接收 100 个字节;而不能循环调用 10 次 recvfrom, 每次接收 10 个字。

UDP的缓冲区

  • UDP 没有真正意义上的 发送缓冲区. 调用 sendto 会直接交给内核, 由内核将数据传给网络层协议进行后续的传输动作;
  • UDP 具有接收缓冲区. 但是这个接收缓冲区不能保证收到的 UDP 报的顺序和发送 UDP 报的顺序一致; 如果缓冲区满了, 再到达的 UDP 数据就会被丢弃;
  • UDP 的 socket 既能读, 也能写, 这个概念叫做 全双工。

UDP的注意事项

Udp协议首部中有一个16位的最大长度,也就是说一个Udp能传输的数据报最大长度是64K(包含Udp首部)。

如果需要传输的数据超过64K,则需要在应用层手动分包,并在接收端手动拼接。

UDP深刻理解

UDP报头:

报头就是结构体变量。

struct udphdr hdr= {...}

image-20241005165034594

UDP报文:

os内可能同时存在大量的报文,一部分正在被向上交付,另一部分正在被向下交付。

所以os要对报文进行各种管理!先描述再组织。

对报文的修改就是对struct sk_buff双向链表的增删查改。

/********************************************
struct sk_buff中,
char *head指向的是数据包的起始位置
char *data指向的是数据包中有效数据的位置

每一个sk_buff都会有一个缓冲区,初始时head不会像常规一样直接指向最开始,而是会指向中间。
********************************************/

image-20241005165114427

三、编写代码

UDP.hpp

// 负责服务端UDP的操作
#pragma once 
#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

enum ErrorCode
{
    SUCCESS,
    FAIL
};
class UDP
{
public:
    UDP(int port)
        :m_ip("0")
        ,m_port(port)
    {}
    void Init()
    {
        // 创建套接字
        m_sockfd = socket(AF_INET, SOCK_DGRAM, 0);
        if (m_sockfd < 0)
        {
            std::cout << "创建套接字失败" << std::endl;
            exit(FAIL);
        }
        std::cout << "创建套接字成功" << std::endl;
        // 绑定套接字
        sockaddr_in addr;
        addr.sin_family = AF_INET;
        addr.sin_port = htons(m_port);
        addr.sin_addr.s_addr = inet_addr(m_ip.c_str());
        int n = bind(m_sockfd, (struct sockaddr *)&addr, sizeof(addr));
        if (n < 0)
        {
            std::cout << "绑定套接字失败" << std::endl;
            exit(FAIL);
        }
        std::cout << "绑定套接字成功" << std::endl;
    }
    void Start()
    {
        while (true)
        {
            // 接收消息
            char buffer[1024];
            sockaddr_in recv_addr;
            socklen_t len = sizeof(recv_addr);
            ssize_t n = recvfrom(m_sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&recv_addr, &len);
            if (n > 0)
            {
                buffer[n] = 0;
                std::cout << "接收成功: " << buffer << std::endl;
            }
            else if (n == 0)
            {
                std::cout << "recvfrom 被关闭" << std::endl;
                exit(SUCCESS);
            }
            else 
            {
                std::cout << "接收异常" << std::endl;
                exit(FAIL);
            }
            // 发送消息
            std::string echo = "Echo #";
            echo += buffer;
            ssize_t m = sendto(m_sockfd, echo.c_str(), echo.size(), 0, (struct sockaddr *)&recv_addr, len);
            if (m < 0)
            {
                std::cout << "发送异常" << std::endl;
                exit(FAIL);
            }
            std::cout << "发送成功" << std::endl;
        }
    }
    ~UDP()
    {}
private:
    std::string m_ip;
    int m_port;
    int m_sockfd;
};

client.cc

#include <iostream>
#include <memory>
#include "UDP.hpp"

// ./client 127.0.0.1 8080
int main(int argc, char *argv[])
{
    if (argc != 3)
    {
        std::cout << "指令错误" << std::endl;
        exit(FAIL);
    }

    std::string ip = argv[1];
    int port = std::stoi(argv[2]);

    // 创建套接字
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0)
    {
        std::cout << "创建套接字失败" << std::endl;
        exit(FAIL);
    }
    std::cout << "创建套接字成功" << std::endl;

    // 读写
    while (true)
    {
        // 发送消息
        std::cout << "Enter #";
        std::string enter;
        std::getline(std::cin, enter);
        sockaddr_in send_addr;
        send_addr.sin_family = AF_INET;
        send_addr.sin_port = htons(port);
        inet_pton(AF_INET, ip.c_str(), &send_addr.sin_addr);
        socklen_t len = sizeof(send_addr);
        ssize_t n = sendto(sockfd, enter.c_str(), enter.size(), 0, (struct sockaddr *)&send_addr, len);
        if (n < 0)
        {
            std::cout << "发送异常" << std::endl;
            exit(FAIL);
        }
        std::cout << "发送成功" << std::endl;
        // 接收消息
        char buffer[1024];
        sockaddr_in addr;
        socklen_t length;
        ssize_t m = recvfrom(sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&addr, &length);
        if (m > 0)
        {
            buffer[n] = 0;
            std::cout << "接收成功: " << buffer << std::endl;
        }
        else if (m == 0)
        {
            std::cout << "recvfrom 被关闭" << std::endl;
            exit(SUCCESS);
        }
        else
        {
            std::cout << "接收异常" << std::endl;
            exit(FAIL);
        }
    }
    return 0;
}

Server.cc

#include <iostream>
#include <memory>
#include "UDP.hpp"

// ./server 8080
int main(int argc, char *argv[])
{
    if (argc != 2)
    {
        std::cout << "指令错误" << std::endl;
        exit(FAIL);
    }
    // 对服务端套接字进行处理
    int port = std::stoi(argv[1]);
    std::unique_ptr<UDP> up = std::make_unique<UDP>(port);
    up->Init();
    up->Start();

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2190909.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

双指针:滑动窗口

题目描述 给定两个字符串 S 和 T&#xff0c;求 S 中包含 T 所有字符的最短连续子字符串的长度&#xff0c;同时要求时间复杂度不得超过 O(n)。 输入输出样例 输入是两个字符串 S 和 T&#xff0c;输出是一个 S 字符串的子串。样例如下&#xff1a; 在这个样例中&#xff0c…

计算机网络:物理层 —— 物理层下的传输媒体

文章目录 传输媒体导向性媒体同轴电缆双绞线光纤光纤分类中心波长光纤规格光纤的优缺点 非导向性媒体ISM 频段无线电波微波激光红外线可见光 传输媒体 传输媒体是计算机网络设备之间的物理通路&#xff0c;也称为传输介质或传输媒介&#xff0c;并不包含在计算机网络体系结构中…

408笔记|随笔记录|自用|2

文章目录 cache和TLB查找总结 接上篇选择题 打开同一个文件共享同一个内存索引结点 cache和TLB cache由SRAM组成&#xff0c;TLB通常由相联存储器组成&#xff0c;可以由SRAM组成。 DRAM需要不断刷新&#xff0c;性能比较低。 TLB缺失可以由软件或者硬件处理&#xff0c;cac…

JC系列CAN通信说明

目录 一、CAN协议二、指令格式三、通信接线3.1、一对一通信3.2、组网通信 四、寄存器定义五、指令说明4、读取电源电压5、读取母线电流6、读取实时速度8、读取实时位置10、读取驱动器温度11、读取电机温度12、读取错误信息32、设定电流33、设定速度35、设定绝对位置37、设定相对…

Java-进阶二

ArrayList的源代码分析&#xff08;扩容原理&#xff09; 1 使用空参构造的集合&#xff0c;在底层创建一个容量为0的数组。2 添加第一个元素时&#xff0c;底层会扩容创建一个容量为10的数组。3 存满时会扩容1.5倍。4 如果一次添加多个元素&#xff0c;1.5倍还放不下&#xff…

机器学习/数据分析--用通俗语言讲解时间序列自回归(AR)模型,并用其预测天气,拟合度98%+

时间序列在回归预测的领域的重要性&#xff0c;不言而喻&#xff0c;在数学建模中使用及其频繁&#xff0c;但是你真的了解ARIMA、AR、MA么&#xff1f;ACF图你会看么&#xff1f;&#xff1f; 时间序列数据如何构造&#xff1f;&#xff1f;&#xff1f;&#xff0c;我打过不少…

提升客户满意度,数字化售后工单系统的实施策略

售后工单系统助企业优化运营&#xff0c;提升客户满意度。涵盖技术支持、维修服务、客户反馈管理等应用场景。ZohoDesk凭高效定制成企业首选。未来趋势包括AI、物联网、AR/VR技术集成&#xff0c;提升服务智能化水平。 一、售后工单系统的重要性 售后工单系统是企业用来管理和…

基于Java的GeoTools对Shapefile文件属性信息深度解析

目录 前言 一、Shapefile的属性列表信息 1、属性表格信息 2、属性表格包含的要素 二、GeoTools对属性表格的解析 1、常规解析方法 2、基于dbf文件的属性信息读取 三、总结 前言 ESRI Shapefile&#xff08;shp&#xff09;&#xff0c;或简称shapefile&#xff0c;是美…

SQL优化 - 排序

文章目录 排序和索引降序索引 FilesortORDER BY 顺序问题ORDER BY LIMIT 排序和索引 如果ORDER BY操作使用了索引&#xff0c;那么就可以避免排序操作&#xff0c;因为索引本身就是按索引 key 排好序的。那什么情况下&#xff0c;ORDER BY会走索引呢&#xff1f; 例如&#…

专题十_穷举vs暴搜vs深搜vs回溯vs剪枝_二叉树的深度优先搜索_算法专题详细总结

目录 搜索 vs 深度优先遍历 vs 深度优先搜索 vs 宽度优先遍历 vs 宽度优先搜索 vs 暴搜 1.深度优先遍历 vs 深度优先搜索(dfs) 2.宽度优先遍历 vs 宽度优先搜索(bfs) 2.关系图暴力枚举一遍所有的情况 3.拓展搜索问题全排列 决策树 1. 计算布尔⼆叉树的值&#xff08;medi…

CF2013E Prefix GCD

【题目大意】 给定一个长度为 n n n 的数列 a 1 … n a_{1 \dots n} a1…n​&#xff0c;你可以将 a 1 … n a_{1 \dots n} a1…n​ 按照任意顺序进行重排&#xff0c;使得&#xff1a; ∑ i 1 n gcd ⁡ { a 1 , a 2 , a 3 , … , a n } \sum\limits_{i1}^{n}\gcd\left \{…

10.5学习

1.GateWay GateWay⽬标是取代Netflflix Zuul&#xff0c;它基于Spring5.0SpringBoot2.0WebFlux等技术开发&#xff0c;提供统⼀的路由⽅式&#xff08;反向代理&#xff09;并且基于 Filter(定义过滤器对请求过滤&#xff0c;完成⼀些功能) 链的⽅式提供了⽹关基本的功能&…

探索 Python 虚拟环境的奥秘:virtualenv 的魔法世界

文章目录 探索 Python 虚拟环境的奥秘&#xff1a;virtualenv 的魔法世界背景&#xff1a;为何选择 virtualenv&#xff1f;虚拟环境的守护者&#xff1a;virtualenv 是什么&#xff1f;安装 virtualenv&#xff1a;简单几步&#xff0c;开启隔离之旅掌握 virtualenv 的基本用法…

Relu激活

ReLU&#xff08;Rectified Linear Unit&#xff09;激活函数 是卷积神经网络&#xff08;CNN&#xff09;以及许多深度学习模型中最常用的激活函数之一。它的主要作用是引入非线性&#xff0c;使模型能够学习和表达更复杂的特征。以下是对ReLU激活函数的详细解释。 1. ReLU的…

C语言进阶版第16课—自定义类型:结构体

文章目录 1. 结构体类型的声明和初始化2. 结构体自引用3. 结构体内存对齐3.1 结构体内存对齐规则3.2 修改默认对齐数 4. 结构体传参4. 结构体实现位段5. 位段使用的注意事项 1. 结构体类型的声明和初始化 结构体在使用之前都要对其类型进行声明&#xff0c;关键字是struct&…

15分钟学 Python 第36天 :Python 爬虫入门(二)

Python 爬虫入门&#xff1a;环境准备 在进行Python爬虫的学习和实践之前&#xff0c;首先需要准备好合适的开发环境。本节将详细介绍Python环境的安装、必要库的配置、以及常用工具的使用&#xff0c;为后续的爬虫编写奠定坚实的基础。 1. 环境准备概述 1.1 为什么环境准备…

mp4转gif在线转换怎么转?7个视频转动图方法不容错过!(超简单)

mp4转gif在线转换怎么转&#xff1f;如今&#xff0c;将mp4视频转换为gif动图格式&#xff0c;满足了人们对易于分享和网络传播内容的需求。与mp4视频相比&#xff0c;gif动图文件体积相对较小&#xff0c;几乎所有网络平台都支持这种格式&#xff0c;无需额外安装插件或软件。…

CSID-GAN:基于生成对抗网络的定制风格室内平面设计框架论文阅读

CSID-GAN: A Customized Style Interior Floor Plan Design Framework Based on Generative Adversarial Network 摘要前言II. CSID-GAN METHODA. Overall FrameworkB. Algorithm and Loss Function III. DATASETS AND EVALUATION METRICSA. DatasetsB. Evaluation Metrics IV.…

信息安全工程师(34)访问控制模型

前言 访问控制模型是实现访问控制的基础&#xff0c;不同的访问控制模型提供了不同的访问控制策略和机制&#xff0c;以适应不同的安全需求。 一、自主访问控制模型&#xff08;DAC&#xff09; 定义&#xff1a;指资源的所有者有权决定谁可以访问其资源以及访问的方式。资源的…

如何实现事件流操作

文章目录 1 概念介绍2 使用方法3 示例代码我们在上一章回中介绍了通道相关的内容,本章回中将介绍StreamProvider组件.闲话休提,让我们一起Talk Flutter吧。 1 概念介绍 在Flutter中Stream是经常使用的组件,对该组件的监听可void main() {///让状态栏和程序的appBar融为一体…