select() 返回时没有传入连接

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

我正在编写一个非常简单的服务器应用程序,只是为了测试一些代码。 创建套接字并将其连接到本地主机和某个端口后,我想使用

bind()
来了解传入连接何时到达绑定套接字。之后,应用程序应将消息打印到一定长度,然后
select()

我的问题基本上是,当我只期望一个连接时,是否需要使用

exit()

listen()
(请记住这只是为了测试)。我相信在这种情况下不需要这些函数,只需要接受多个传入请求。难道我错了?

考虑到上述想法,我编写了以下代码

accept()

TCPcreate()

int main() { int fd = TCPcreate(atoh("127.0.0.1"), 15000); /*my localhost address*/ char *str = malloc(100); int a; fd_set rfds; FD_ZERO(&rfds); FD_SET(fd,&rfds); a = select(fd+1,&rfds,(fd_set*)NULL,(fd_set*)NULL,(struct timeval*)NULL); // printf("select returns %d\nfd = %d\n", a, fd); // printf("fd is set? %s\n", FD_ISSET(fd,&rfds) ? "yes" : "no"); a = TCPrecv(fd, str, 100); /*receive at most 100B */ // printf("%d\n", a); printf("%s\n", str); close(fd); exit(0); }

int TCPcreate(unsigned long IP, unsigned short port)
{
    int fd;
    struct sockaddr_in address;

    fd = socket(AF_INET, SOCK_STREAM, 0);
    if(fd==-1)
    {
        return -1;
    }

    memset(&address, 0, sizeof(address));
    address.sin_family = AF_INET;
    address.sin_addr.s_addr = htonl(IP);
    address.sin_port = htons(port);

    /* struct sockaddr_in is the same size as struct sockaddr */
    if(bind(fd, (struct sockaddr*)&address, sizeof(address))==-1)
    {
        return -2;
    }

    return fd;
}

只是以主机字节顺序返回其参数。


当我运行该程序时,发生的情况是

atoh()

不会阻止等待连接。相反,它立即返回 1。如果我取消注释

select()
,我得到的是

printf()

我在这里缺少什么?...

c sockets network-programming posix-select
2个回答
1
投票

select returns 1 fd = 3 is set? yes -1 (blank line)

 的 POSIX 规范,返回的文件描述符已准备好读取、写入,或者有错误条件。这不会将“
select()
 会成功的套接字”列为可检测条件之一。因此,您需要使用 
listen()
listen()
;只有在接受连接后才能在描述符上使用 
accept()

正如

Gonçalo Ribeiro

所指出的,select() 的规范还指出:


如果套接字当前正在侦听,则如果已收到传入连接请求,则应将其标记为可读,并且对
select()

函数的调用应在不阻塞的情况下完成。



这意味着您必须在绑定的套接字上执行
accept()

,但您可以在多个套接字上等待传入连接。

    


0
投票

选择的问题在于您的代码中 - 将选择保留在循环中。由于它是非阻塞调用,因此它只会检查是否有人在听。所以,你可以使用循环来多次检查监听。

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