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

时间:2014-03-02 16:34:33

标签: c sockets select network-programming

我正在编写一个非常简单的服务器应用程序,仅用于测试一些代码。 在创建套接字并bind()到我的localhost和某个端口后,我想使用select()来知道传入连接何时到达绑定套接字。之后,应用程序应该将消息打印到某个长度,然后exit()

我的问题基本上是在我期待只有一个连接时需要使用listen()accept()(请记住这仅用于测试)。我相信在这种情况下不需要这些功能,只需要接受多个传入请求。我错了吗?

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

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);
}

TCPcreate()

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()只是以主机字节顺序返回其参数。

运行程序时会发生什么,select()不会阻止等待连接。相反,它立即返回1.如果我取消注释printf() s得到的是

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

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

2 个答案:

答案 0 :(得分:1)

如果查看select()的POSIX规范,则返回的文件描述符已准备好进行读取,写入或出现错误。这不列出'listen()将成功的套接字作为可检测条件之一。因此,您需要使用listen() accept() ;只有在您接受连接后,才能在描述符上使用select()

正如Gonçalo Ribeiro所述,select()的规范也说明了:

  

如果套接字当前正在侦听,那么如果收到传入的连接请求,则应将其标记为可读,并且对accept()函数的调用应完成而不会阻塞。

这意味着您必须在绑定套接字上完成listen(),但您可以在多个套接字上等待传入连接。

答案 1 :(得分:0)

如果你想要阻止呼叫 - 请使用listen()。

选择的问题在于你的代码是 - 保持选择在循环中。因为它是一个非阻塞呼叫,它只会检查一个人是否在那里听。因此,您可以使用循环检查多次收听。