非阻塞udp套接字recvfrom未接收

问题描述 投票:0回答:1

我有一项作业要求我设置一个 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;
}

如果我们让套接字阻塞,这个例子就可以正常工作。 我的猜测是,关于非阻塞套接字的某些内容我不理解,并且搜索它还没有引导我到任何地方。我希望你能帮助我!

c sockets nonblocking recvfrom
1个回答
0
投票

正如 @SteffenUllrich 指出的,在检查

errno
之前,我们必须确保
recvfrom
的返回值不好。

int rv = recvfrom(..);
if (rv < 0 && errno == EWOULDBLOCK) { ... }

愚蠢的错误,我的错。

© www.soinside.com 2019 - 2024. All rights reserved.