Linux UDP(用户数据报协议)是一种无连接的、不可靠的传输层协议,适用于对实时性要求高但对数据完整性要求相对较低的应用场景,如视频流、在线游戏、VoIP(语音通话)等。以下是UDP传输数据的图解及相关概念:
+-------------------+ +-------------------+ +-------------------+
| 发送方应用 | -----> | 发送方UDP套接字 | -----> | 网络传输层 |
+-------------------+ +-------------------+ +-------------------+
|
v
+-------------------+ +-------------------+ +-------------------+
| 接收方UDP套接字 | <----- | 网络传输层 | <----- | 接收方应用 |
+-------------------+ +-------------------+ +-------------------+
原因:UDP是无连接的,不保证数据报的顺序和可靠性。
解决方法:
// 示例代码:发送端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define PORT 8080
#define BUFFER_SIZE 1024
int main() {
int sockfd;
struct sockaddr_in servaddr, cliaddr;
char buffer[BUFFER_SIZE];
// 创建UDP套接字
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
perror("socket creation failed");
exit(EXIT_FAILURE);
}
memset(&servaddr, 0, sizeof(servaddr));
memset(&cliaddr, 0, sizeof(cliaddr));
// 填充服务器信息
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = INADDR_ANY;
servaddr.sin_port = htons(PORT);
// 绑定套接字
if (bind(sockfd, (const struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) {
perror("bind failed");
exit(EXIT_FAILURE);
}
// 发送数据
sendto(sockfd, (const char *)buffer, strlen(buffer), MSG_CONFIRM, (const struct sockaddr *)&cliaddr, sizeof(cliaddr));
printf("Message sent.\n");
close(sockfd);
return 0;
}
// 示例代码:接收端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define PORT 8080
#define BUFFER_SIZE 1024
int main() {
int sockfd;
struct sockaddr_in servaddr, cliaddr;
socklen_t len;
char buffer[BUFFER_SIZE];
// 创建UDP套接字
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
perror("socket creation failed");
exit(EXIT_FAILURE);
}
memset(&servaddr, 0, sizeof(servaddr));
memset(&cliaddr, 0, sizeof(cliaddr));
// 填充服务器信息
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = INADDR_ANY;
servaddr.sin_port = htons(PORT);
// 绑定套接字
if (bind(sockfd, (const struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) {
perror("bind failed");
exit(EXIT_FAILURE);
}
len = sizeof(cliaddr);
// 接收数据
recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, 0, (struct sockaddr *)&cliaddr, &len);
printf("Received message: %s\n", buffer);
close(sockfd);
return 0;
}
参考链接:
通过以上图解和示例代码,您可以更好地理解Linux UDP如何传输数据,并解决常见的数据丢失或乱序问题。
领取专属 10元无门槛券
手把手带您无忧上云