[C++/Linux] UDP编程

一. UDP函数

     UDP(用户数据报协议,User Datagram Protocol)是一种无连接的网络协议,用于在互联网上交换数据。它允许应用程序发送数据报给另一端的应用程序,但不保证数据报能成功到达,也就是说,它提供的是一种不可靠的服务。与TCP(传输控制协议)相比,UDP的优势在于它的简单性和低开销,使得它特别适合于那些对实时性要求高的应用,如视频会议和在线游戏。

在编程中,使用UDP进行数据通信涉及到一系列的函数调用。这些函数调用通常通过操作系统提供的网络接口与内核进行交互。下面是一些在UDP编程中常用的函数及其与内核的关系:

1. socket()

  • 用途:创建一个新的套接字(socket),它是进行网络通信的基础。
  • 与内核的关系:调用这个函数会让操作系统内核创建一个套接字数据结构,并返回一个引用这个结构的文件描述符(FD)。此后,应用程序通过这个文件描述符进行所有网络操作。

2. bind()

  • 用途:将套接字与一个特定的IP地址和端口号绑定。对于服务器端应用程序来说,这样做可以让它在特定的端口上监听来自客户端的连接或数据。
  • 与内核的关系:此函数将应用程序指定的IP地址和端口号信息注册到内核中,内核会用这些信息来识别到来的数据包是属于哪个应用程序的。

3. sendto()

  • 用途:用于向特定的目标地址发送数据。在UDP编程中,每次发送数据时都需要指定目的地的地址和端口号。
  • 与内核的关系:应用程序调用sendto()函数后,数据和目标地址信息被传递给内核,内核负责将数据打包成UDP数据报,并通过网络发送出去。

4. recvfrom()

  • 用途:用于接收来自任何地址的数据。应用程序可以通过这个函数获取发送方的地址信息。
  • 与内核的关系:当UDP数据报到达操作系统时,内核会根据数据报的目的端口号找到对应的套接字,并将数据放入该套接字的接收缓冲区。应用程序通过recvfrom()函数从这个缓冲区读取数据。

5. close()

  • 用途:关闭套接字,释放与之相关的资源。
  • 与内核的关系:调用close()函数后,应用程序通知内核它已经完成了对该套接字的使用,内核随后会清理与这个套接字相关的资源。

二. TCP编程与UDP编程的区别:

  1. 连接性

    • TCP:面向连接的协议,通信双方需要建立一个稳定的连接,然后才能进行数据传输。
    • UDP:无连接的协议,数据可以直接发送给接收方,不需要建立连接。
  2. 可靠性

    • TCP:提供可靠的数据传输,通过序列号、确认应答、重传机制等确保数据的可靠到达。
    • UDP:不保证数据可靠到达,可能会出现丢包,但UDP在丢包处理上更简单,不进行重传。
  3. 速度

    • TCP:由于建立连接和保证数据可靠,速度相对较慢。
    • UDP:因为没有连接建立和数据可靠性检查,速度相对较快。
  4. 数据流

    • TCP:保证数据的顺序,确保应用层接收到的数据顺序与发送顺序一致。
    • UDP:不保证数据顺序,数据可能会乱序到达。
  5. 使用场景

    • TCP:适用于对数据准确性要求较高的场景,如文件传输、邮件传输等。
    • UDP:适用于对速度要求较高,但可以容忍一定程度数据丢失的场景,如视频会议、在线游戏等。

三. UDP代码例子:

服务器代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>#define PORT 8080
#define BUFFER_SIZE 1024// 函数声明
void error_handling(char *message);int main(int argc, char *argv[]) {int server_socket;struct sockaddr_in server_address, client_address;socklen_t client_address_size = sizeof(client_address);char message[BUFFER_SIZE];// 创建UDP套接字server_socket = socket(PF_INET, SOCK_DGRAM, 0);if (server_socket == -1) {error_handling("socket() error");}// 设置服务器地址结构memset(&server_address, 0, sizeof(server_address));server_address.sin_family = AF_INET;server_address.sin_addr.s_addr = htonl(INADDR_ANY);server_address.sin_port = htons(PORT);// 绑定套接字到端口if (bind(server_socket, (struct sockaddr *)&server_address, sizeof(server_address)) == -1) {error_handling("bind() error");}// 服务器主循环while (1) {// 接收客户端发送的数据int str_len = recvfrom(server_socket, message, BUFFER_SIZE, 0, (struct sockaddr *)&client_address, &client_address_size);if (str_len == -1) {error_handling("recvfrom() error");}// 打印接收到的消息printf("Received from client: %s\n", message);// 处理消息(这里简单地转换为大写)for (int i = 0; i < str_len; i++) {message[i] = toupper(message[i]);}// 发送处理后的消息回客户端sendto(server_socket, message, str_len, 0, (struct sockaddr *)&client_address, client_address_size);}// 关闭套接字close(server_socket);return 0;
}void error_handling(char *message) {fputs(message, stderr);fputc('\n', stderr);exit(1);
}

客户端代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>#define PORT 8080
#define BUFFER_SIZE 1024// 函数声明
void error_handling(char *message);int main(int argc, char *argv[]) {int client_socket;struct sockaddr_in server_address;char message[BUFFER_SIZE];int message_len;// 创建UDP套接字client_socket = socket(PF_INET, SOCK_DGRAM, 0);if (client_socket == -1) {error_handling("socket() error");}// 设置服务器地址结构memset(&server_address, 0, sizeof(server_address));server_address.sin_family = AF_INET;server_address.sin_addr.s_addr = inet_addr("127.0.0.1");server_address.sin_port = htons(PORT);// 循环读取用户输入并发送给服务器while (1) {printf("Input message(Q to quit): ");fgets(message, BUFFER_SIZE, stdin);// 检查是否退出if (!strcmp(message, "q\n") || !strcmp(message, "Q\n")) {break;}// 发送消息给服务器sendto(client_socket, message, strlen(message), 0, (struct sockaddr *)&server_address, sizeof(server_address));// 接收服务器响应message_len = recvfrom(client_socket, message, BUFFER_SIZE, 0, NULL, 0);if (message_len == -1) {error_handling("recvfrom() error");}// 打印服务器响应message[message_len] = 0;printf("Message from server: %s\n", message);}// 关闭套接字close(client_socket);return 0;
}void error_handling(char *message) {fputs(message, stderr);fputc('\n', stderr);exit(1);
}

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

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

相关文章

双指针2s总结

5.双指针 双指针理论基础 那么vector< char > 和 string 又有什么区别呢&#xff1f; 其实在基本操作上没有区别&#xff0c;但是 string提供更多的字符串处理的相关接口&#xff0c;例如string 重载了&#xff0c;而vector却没有。 所以想处理字符串&#xff0c;我们…

Java常用类(二)

常用类&#xff08;二&#xff09; Object类 超类&#xff0c;基类&#xff0c;所有类的直接或间接父类&#xff0c;位于继承树的最高层任何类&#xff0c;如没有书写extends显示继承某个类&#xff0c;都默认直接继承Object类&#xff0c;否则为间接继承Object类中所定义的方…

C语言题目:数组寻找最小绝对值

题目描述 输入10个数&#xff0c;找出其中绝对值最小的数&#xff0c;将它和最后一个数交换&#xff0c;然后输出这10个数。 输入格式 十个数 输出格式 交换后的十个数 样例输入 10 2 30 40 50 60 70 80 90 100 样例输出 10 100 30 40 50 60 70 80 90 2 代码解析 包含…

扫描IP开放端口该脚本用于对特定目标主机进行常见端口扫描(加载端口字典)或者指定端口扫描,判断目标主机开

扫描IP开放端口该脚本用于对特定目标主机进行常见端口扫描(加载端口字典)或者指定端口扫描,判断目标主机开 #/bin/bash #该脚本用于对特定目标主机进行常见端口扫描(加载端口字典)或者指定端口扫描,判断目标主机开放来哪些端口 #用telnet方式 IP$1 #IP119.254.3.28 #获得IP的前…

BUUCTF刷题十一道(12)SSTI专题一

文章目录 学习文章[Flask]SSTI[GWCTF 2019]你的名字[第三章 web进阶]SSTI[pasecactf_2019]flask_ssti[NewStarCTF 公开赛赛道]BabySSTI_One[Dest0g3 520迎新赛]EasySSTI[NewStarCTF 公开赛赛道]BabySSTI_Two[NewStarCTF 公开赛赛道]BabySSTI_Three[GYCTF2020]FlaskApp[CSCCTF 2…

网易云歌曲评论抓取

网易云歌曲评论爬取 步骤1.找到一首歌曲2.按下F12键打开开发者模式,对其进行抓包3.查找获得评论数据的接口4.对获得评论数据接口进行分析5.构建加密函数方法一方法二运行结果全部代码使用Js文件只使用python新的代码小结与展望这次的任务是获取网易云音乐下面的评论,涉及的知…

List接口(1) | ArrayList

参考阅读&#xff1a;ArrayList 源码分析 一、特点 ArrayList 继承于 AbstractList &#xff0c;实现了 List, RandomAccess, Cloneable, java.io.Serializable 这些接口&#xff0c;具有以下特点&#xff1a; 支持添加&#xff0c;删除和查找操作支持随机快速访问&#xff…

前端ocr技术:electron+vue3中使用tesseract插件识别图片中字符

同学们可以私信我加入学习群&#xff01; 正文开始 前言一、electron各种csp问题二、试用插件总结 前言 项目需要ocr技术识别图片中的中文字符&#xff0c;本来这部分是后端的工作&#xff0c;但是因为各种原因&#xff0c;决定前端也做一个版本。 在ai时代之前&#xff0c;o…

基于spring boot的漫画之家系统

基于spring boot的漫画之家系统设计与实现 开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09; 数据库工具&#xff1a;Navicat11 开发软件&…

pta L1-012 计算指数

L1-012 计算指数 分数 5 全屏浏览 切换布局 作者 陈越 单位 浙江大学 真的没骗你&#xff0c;这道才是简单题 —— 对任意给定的不超过 10 的正整数 n&#xff0c;要求你输出 2n。不难吧&#xff1f; 输入格式&#xff1a; 输入在一行中给出一个不超过 10 的正整数 n。 …

Java排序算法

冒泡排序 相邻两个元素比较&#xff0c;每次排序获得一个当前排序区间的最大值 // 冒泡排序 // 外层循环每次最后一个元素已经是最大的了&#xff0c;所以每次循环都减少一个元素 for (int i arr.length - 1; i >0; i--) {// 内层循环每次都是从第一个元素开始&#xff0c;…

鲁棒线性模型估计(Robust linear model estimation)

鲁棒线性模型估计 1.RANSAC算法1.1 算法的基本原理1.2 迭代次数N的计算1.3 参考代码 参考文献 当数据中出现较多异常点时&#xff0c;常用的线性回归OLS会因为这些异常点的存在无法正确估计线性模型的参数&#xff1a; W ( X T X ) − 1 X T Y \qquad \qquad W(X^TX)^{-1}X^T…

力扣热题100_链表_19_删除链表的倒数第 N 个结点

文章目录 题目链接解题思路解题代码 题目链接 19. 删除链表的倒数第 N 个结点 给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], n 2 输出&#xff1a;[1,2,3,5] 示例 2&am…

数据结构—图

图的基本概念 图就是由顶点的有穷非空集合和顶点之间的边组成的集合。通常表示为&#xff1a;G(V,E)&#xff0c;其中&#xff0c;G 表示一个图&#xff0c;V 表示顶点的集合&#xff0c;E 表示边的集合。 顶点 图中的数据元素&#xff0c;我们称之为顶点&#xff0c;图至少有…

2024年3月电子学会青少年软件编程 中小学生Python编程等级考试一级真题解析(判断题)

2024年3月Python编程等级考试一级真题解析 判断题&#xff08;共10题&#xff0c;每题2分&#xff0c;共20分&#xff09; 26、turtle 画布的坐标系原点是在画布的左上角 答案&#xff1a;错 考点分析&#xff1a;考查turtle相关知识&#xff0c;turtle画布坐标系是在画布的…

KNN分类算法的MATLAB实现以及可视化

一、KNN简介 KNN算法&#xff0c;即K-Nearest Neighbors&#xff0c;是一种常用的监督学习算法&#xff0c;可以用于分类问题&#xff0c;并且在实际应用中取得了广泛的成功。 二、KNN算法的基本原理 对于给定的测试样本&#xff0c;KNN算法首先计算它与训练集中所有样本的距…

服务器硬件基础知识解析

导言 在当今信息化时代&#xff0c;服务器扮演着至关重要的角色&#xff0c;它们是存储、处理和传输数据的关键设备。本文将介绍服务器硬件的基础知识&#xff0c;包括服务器的组成部分、硬件选型和性能评估等内容&#xff0c;旨在帮助读者更好地理解和应用服务器技术。 服务…

【数据库】规划与分析

目录 1.规划与分析 的步骤&#xff08;3步&#xff09;&#xff1f; 2.系统规划与定义 的步骤&#xff08;4步&#xff09;&#xff1f; 3.可行性分析 的步骤&#xff08;4步&#xff09;&#xff1f;&#xff08;-表示包括 、和/表示并列&#xff09; 4.项目规划 的步…

Vue - 你知道Vue2中对象动态新增属性,视图无法更新的原因吗

难度级别:中高级及以上 提问概率:55% 这道题面试官会这样描述,比如有这样一个场景,一个对象里有name属性,可以正常显示在页面中。但后续动态添加了一个age属性,通过调试打印发现对象里的age属性已经添加了上了,但试图中却没有展示出来,…

Axure案例分享—垂直手风琴(附下载地址)

今天分享的案例是Axure8(兼容9和10)制作的垂直手风琴 一、功能介绍 折叠或展开多个面板内容&#xff0c;默认为展开一项内容&#xff0c;点击任一收起的选项&#xff0c;展开面板&#xff0c;其他面板收起二、制作过程 原型是由矩形组件以及动态面板构成&#xff0c; 拖入一…