UDP 与 TCP 的核心区别
-
无连接:不需要建立/维护连接
-
不可靠:不保证数据包顺序和到达
-
高效:头部开销小,没有连接管理负担
-
支持广播/多播:可以向多个目标同时发送数据
一、基础UDP服务器实现
1. 创建 UDP 套接字
#include <event2/event.h>
#include <event2/listener.h>
#include <event2/bufferevent.h>
#include <arpa/inet.h>int create_udp_socket(int port) {int fd = socket(AF_INET, SOCK_DGRAM, 0);if (fd < 0) {perror("socket");return -1;}struct sockaddr_in sin;memset(&sin, 0, sizeof(sin));sin.sin_family = AF_INET;sin.sin_port = htons(port);sin.sin_addr.s_addr = htonl(INADDR_ANY);if (bind(fd, (struct sockaddr*)&sin, sizeof(sin)) < 0) {perror("bind");close(fd);return -1;}// 设置非阻塞evutil_make_socket_nonblocking(fd);return fd;
}
2. UDP 事件处理
void udp_read_cb(evutil_socket_t fd, short events, void *arg) {struct sockaddr_in client_addr;socklen_t addr_len = sizeof(client_addr);char buf[1500]; // 标准MTU大小ssize_t n;while ((n = recvfrom(fd, buf, sizeof(buf), 0,(struct sockaddr*)&client_addr, &addr_len)) > 0) {// 处理接收到的数据printf("Received %zd bytes from %s:%d\n", n, inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));// 简单回显sendto(fd, buf, n, 0, (struct sockaddr*)&client_addr, addr_len);}if (n < 0 && errno != EAGAIN && errno != EWOULDBLOCK) {perror("recvfrom");}
}int main() {int udp_fd = create_udp_socket(8080);if (udp_fd < 0) return 1;struct event_base *base = event_base_new();struct event *udp_event = event_new(base, udp_fd, EV_READ | EV_PERSIST, udp_read_cb, NULL);event_add(udp_event, NULL);printf("UDP server started on port 8080\n");event_base_dispatch(base);event_free(udp_event);close(udp_fd);event_base_free(base);return 0;
}
3. UDP 服务器示例代码
#include <event2/event.h>
#include <event2/bufferevent.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>
#include <time.h>#define MAX_CLIENTS 10000
#define MAX_PACKET_SIZE 1500struct udp_client {struct sockaddr_in addr;time_t last_active;
};struct udp_server {struct event_base *base;int sockfd;struct event *ev;struct udp_client clients[MAX_CLIENTS];
};void udp_read_cb(evutil_socket_t fd, short events, void *arg) {struct udp_server *server = arg;char buf[MAX_PACKET_SIZE];struct sockaddr_in client_addr;socklen_t addr_len = sizeof(client_addr);ssize_t n = recvfrom(fd, buf, sizeof(buf), 0,(struct sockaddr*)&client_addr, &addr_len);if (n <= 0) return;// 查找或创建客户端记录struct udp_client *client = NULL;for (int i = 0; i < MAX_CLIENTS; i++) {if (memcmp(&server->clients[i].addr, &client_addr, addr_len) == 0) {client = &server->clients[i];break;}}if (!client) {// 查找空闲槽位for (int i = 0; i < MAX_CLIENTS; i++) {if (server->clients[i].last_active == 0) {client = &server->clients[i];memcpy(&client->addr, &client_addr, addr_len);break;}}}if (client) {client->last_active = time(NULL);// 业务处理printf("Received %zd bytes from %s:%d\n", n, inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));// 回显sendto(fd, buf, n, 0, (struct sockaddr*)&client_addr, addr_len);} else {printf("Client limit reached, dropping packet\n");}
}struct udp_server* udp_server_create(int port) {struct udp_server *server = malloc(sizeof(struct udp_server));memset(server, 0, sizeof(*server));// 创建socketserver->sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (server->sockfd < 0) {perror("socket");free(server);return NULL;}// 绑定地址struct sockaddr_in sin;memset(&sin, 0, sizeof(sin));sin.sin_family = AF_INET;sin.sin_port = htons(port);sin.sin_addr.s_addr = htonl(INADDR_ANY);if (bind(server->sockfd, (struct sockaddr*)&sin, sizeof(sin)) < 0) {perror("bind");close(server->sockfd);free(server);return NULL;}// 设置非阻塞evutil_make_socket_nonblocking(server->sockfd);// 创建事件基server->base = event_base_new();if (!server->base) {close(server->sockfd);free(server);return NULL;}// 创建事件server->ev = event_new(server->base, server->sockfd, EV_READ | EV_PERSIST, udp_read_cb, server);event_add(server->ev, NULL);return server;
}void udp_server_run(struct udp_server *server) {event_base_dispatch(server->base);
}void udp_server_free(struct udp_server *server) {if (!server) return;if (server->ev) event_free(server->ev);if (server->base) event_base_free(server->base);if (server->sockfd > 0) close(server->sockfd);free(server);
}int main() {struct udp_server *server = udp_server_create(8080);if (!server) return 1;printf("UDP server started on port 8080\n");udp_server_run(server);udp_server_free(server);return 0;
}
4. UDP 客户器示例代码
#include <event2/event.h>
#include <event2/dns.h>
#include <event2/bufferevent.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>
#include <pthread.h>
#include <unistd.h>struct udp_client {struct event_base *base;struct evdns_base *dns;int sockfd;struct sockaddr_in server_addr;int connected;pthread_mutex_t lock;
};void udp_read_cb(evutil_socket_t fd, short events, void *arg) {struct udp_client *client = arg;char buf[1500];struct sockaddr_in from_addr;socklen_t addr_len = sizeof(from_addr);ssize_t n = recvfrom(fd, buf, sizeof(buf), 0,(struct sockaddr*)&from_addr, &addr_len);if (n > 0) {buf[n] = '\0';pthread_mutex_lock(&clie