当前位置: 首页 > news >正文

C++网络编程 4.UDP套接字(socket)编程示例程序

以下是基于UDP协议的完整客户端和服务器代码。UDP与TCP的核心区别在于无连接特性,因此代码结构会更简单(无需监听和接受连接)。

UDP服务器代码(udp_server.cpp)

#include <iostream>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <cstring>int main() {// 1. 创建UDP套接字(SOCK_DGRAM表示UDP)int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (sockfd == -1) {std::cerr << "Failed to create socket" << std::endl;return 1;}// 2. 绑定IP和端口struct sockaddr_in server_addr;memset(&server_addr, 0, sizeof(server_addr));server_addr.sin_family = AF_INET;server_addr.sin_addr.s_addr = htonl(INADDR_ANY);  // 监听所有IPserver_addr.sin_port = htons(9888);  // UDP端口if (bind(sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr)) == -1) {std::cerr << "Failed to bind socket" << std::endl;close(sockfd);return 1;}std::cout << "UDP Server started, listening on port 9888..." << std::endl;// 3. 接收客户端数据并回复char buffer[1024];struct sockaddr_in client_addr;socklen_t client_addr_len = sizeof(client_addr);while (true) {  // 循环接收多个客户端消息// 接收客户端数据(自动获取客户端地址)memset(buffer, 0, sizeof(buffer));int recv_len = recvfrom(sockfd, buffer, sizeof(buffer), 0, (struct sockaddr*)&client_addr, &client_addr_len);if (recv_len == -1) {std::cerr << "Failed to receive data" << std::endl;continue;  // 继续接收其他客户端消息}// 打印客户端信息和消息char client_ip[INET_ADDRSTRLEN];inet_ntop(AF_INET, &client_addr.sin_addr, client_ip, INET_ADDRSTRLEN);std::cout << "Received from " << client_ip << ":" << ntohs(client_addr.sin_port) << ": " << buffer << std::endl;// 回复客户端std::string response = "Hello, client! I got your message: " + std::string(buffer);sendto(sockfd, response.c_str(), response.size(), 0, (struct sockaddr*)&client_addr, client_addr_len);}// 4. 关闭套接字(实际不会执行到这里,需按Ctrl+C终止)close(sockfd);return 0;
}

UDP客户端代码(udp_client.cpp)

#include <iostream>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <cstring>
using namespace std;int main() {// 1. 创建UDP套接字int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (sockfd == -1) {std::cerr << "Failed to create socket" << std::endl;return 1;}// 2. 设置服务器地址struct sockaddr_in server_addr;memset(&server_addr, 0, sizeof(server_addr));server_addr.sin_family = AF_INET;server_addr.sin_port = htons(9888);  // 服务器端口inet_pton(AF_INET, "127.0.0.1", &server_addr.sin_addr);  // 服务器IP// 3. 发送消息并接收回复string message = "Hello, UDP server!";sendto(sockfd, message.c_str(), message.size(), 0, (struct sockaddr*)&server_addr, sizeof(server_addr));// 接收服务器回复char buffer[1024];struct sockaddr_in server_response_addr;socklen_t server_addr_len = sizeof(server_response_addr);memset(buffer, 0, sizeof(buffer));int recv_len = recvfrom(sockfd, buffer, sizeof(buffer), 0, (struct sockaddr*)&server_response_addr, &server_addr_len);if (recv_len == -1) {std::cerr << "Failed to receive response" << std::endl;close(sockfd);return 1;}cout << "Received from server: " << buffer << endl;// 4. 关闭套接字close(sockfd);return 0;
}

核心差异对比(TCP vs UDP)

特性TCPUDP
连接方式面向连接(三次握手/四次挥手)无连接(直接发送)
套接字类型SOCK_STREAMSOCK_DGRAM
核心APIbind→listen→accept→connectbind(仅服务器需要)
数据传输函数read/writesend/recvsendto/recvfrom
可靠性可靠(自动重传、按序到达)不可靠(可能丢包、乱序)
通信流程先建立连接,再固定双方通信每次发送需指定目标地址

UDP编程关键点说明

  1. 无连接特性

    • 服务器无需 listen()accept(),直接接收数据;
    • 客户端无需 connect(),直接向服务器地址发送数据。
  2. 核心函数 sendto()recvfrom()

    // 发送数据到指定地址
    ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);// 从指定地址接收数据(自动获取发送方地址)
    ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);
    
    • 每次发送/接收都需指定对方地址(struct sockaddr*),因此UDP可与多个不同目标通信(无固定连接)。
  3. UDP服务器的多客户端处理

    • 单个套接字可处理多个客户端(通过 recvfrom() 获取客户端地址,sendto() 回复特定客户端);
    • 无需为每个客户端创建新套接字(与TCP不同)。
  4. 数据边界

    • UDP是“数据报”协议,发送的数据有明确边界(发送几次就接收几次,不会粘包);
    • 接收缓冲区需足够大,否则会导致数据截断(如发送1000字节,但缓冲区只有512字节,则仅接收前512字节)。

编译和运行步骤

  1. 编译程序:

    g++ udp_server.cpp -o udp_server
    g++ udp_client.cpp -o udp_client
    
  2. 启动服务器:

    ./udp_server
    # 输出:UDP Server started, listening on port 9888...
    
  3. 启动客户端(新终端):

    ./udp_client
    # 输出:Received from server: Hello, client! I got your message: Hello, UDP server!
    
  4. 服务器端会显示:

    Received from 127.0.0.1:xxxx: Hello, UDP server!
    

    xxxx 是客户端随机分配的端口号)

适用场景

  • UDP:适合实时性要求高、允许少量丢包的场景(如视频/语音通话、游戏、实时监控)。
  • TCP:适合可靠性要求高、数据完整性重要的场景(如文件传输、网页浏览、邮件)。
http://www.dtcms.com/a/284775.html

相关文章:

  • UNISOC8850平台Log工具使用说明
  • 基于python和neo4j构建知识图谱医药问答系统
  • Cursor开发步骤
  • 大模型狂想曲:当AI学会“思考”,世界如何被重塑?
  • 用aws下载NOAA的MB文件
  • 【LeetCode 热题 100】230. 二叉搜索树中第 K 小的元素——中序遍历
  • 基于邻域统计分析的点云去噪方法
  • C++ 回调函数全面指南:从基础到高级应用场景实战
  • Junit5
  • 分区表设计:历史数据归档与查询加速
  • ffmpeg转dav为mp4
  • FFmpeg 直播推流
  • 网络编程-java
  • 876. 链表的中间节点
  • CNN(卷积神经网络)--李宏毅deep-learning(起飞!!)
  • MISRA C-2012准则之常量
  • 【Dv3Admin】传递数据实现查询功能
  • RISC-V和ARM有何区别?
  • 学习日志12 python
  • 云原生技术与应用-Kubernetes架构原理与集群环境部署
  • MySQL详解一
  • 【自用】JavaSE--集合框架(一)--Collection集合体系
  • AC7670模块日志的抓取
  • Redis7 底层数据结构解析
  • 【Elasticsearch】Elasticsearch 快照恢复 API 参数详解
  • SmartX 用户建云实践|富士康:基于榫卯企业云平台构建分布式云,支撑全球多地工厂重要产线
  • 百度搜索引擎蜘蛛IP地址段
  • 数据结构排序算法总结(C语言实现)
  • 低功耗、高性能和高度集成的SoC,32位MCU和各种外围IO芯片XL2417D
  • java常见算法合集