我有一项作业要求我设置一个 UDP 服务器,该服务器接受来自多个用户的非阻塞请求,而无需使用
select
或 fork
。
好吧,巧妙的挑战。我的想法:让我们创建一个非阻塞 UDP 套接字并定期接收,因为这正是分配所要求的(服务器定期检查是否有注册请求 [...])。
问题:每次我 recvfrom
即使客户端正在发送消息,我总是得到 EWOULDBLOCK
。如果您对此感兴趣,我写了一个最小的工作示例:
// server.c
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <time.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#define BUFSIZE 1024
#define PORT 12345
int main(int argc, char** argv)
{
char buffer[BUFSIZE];
int rv;
int sock;
struct sockaddr_in addr;
socklen_t addr_len = sizeof(addr);
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = INADDR_ANY;
addr.sin_port = htons(PORT);
sock = socket(AF_INET, SOCK_DGRAM | SOCK_NONBLOCK, 0);
bind(sock, (struct sockaddr*)&addr, addr_len);
while (1)
{
memset(buffer, 0, sizeof(buffer));
rv = recvfrom(sock, buffer, sizeof(buffer), 0, (struct sockaddr*)&addr, &addr_len);
if (errno == EWOULDBLOCK)
{
puts(".");
}
else
{
char ipstr[INET_ADDRSTRLEN];
inet_ntop(AF_INET, &addr, ipstr, addr_len);
buffer[rv] = '\0';
printf("new request from [%s:%d]:\n%s\n", ipstr, htons(addr.sin_port), buffer);
}
sleep(1);
}
return 0;
}
// client.c
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#define BUFSIZE 1024
#define PORT 12345
#define IPADDR "127.0.0.1"
const char* msgtosend = "can you hear me?";
int main(int argc, char** argv)
{
char buffer[BUFSIZE];
int sock;
struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(PORT);
inet_pton(AF_INET, IPADDR, &addr.sin_addr.s_addr);
sock = socket(AF_INET, SOCK_DGRAM, 0);
connect(sock, (struct sockaddr*)&addr, sizeof(addr));
while (1)
{
send(sock, msgtosend, strlen(msgtosend), 0);
puts(".");
sleep(1);
}
return 0;
}
如果我们让套接字阻塞,这个例子就可以正常工作。 我的猜测是,关于非阻塞套接字的某些内容我不理解,并且搜索它还没有引导我到任何地方。我希望你能帮助我!
正如 @SteffenUllrich 指出的,在检查
errno
之前,我们必须确保 recvfrom
的返回值不好。
int rv = recvfrom(..);
if (rv < 0 && errno == EWOULDBLOCK) { ... }
愚蠢的错误,我的错。