什么可以导致select()不返回?

时间:2015-04-03 16:09:44

标签: c sockets posix

我正在编写一个通过Unix Domain Sockets进行通信的客户端和服务器(用于POSIX Development类中的任务,只是为了学习这个概念)。会发生什么是客户端向服务器发送命令,服务器解析它们并以答案响应。我的问题是客户端没有收到服务器的第一个查询,与其他客户端不同。

更清楚的是,客户端没有收到对给定服务器实例的第一个查询,如果我关闭客户端实例并打开一个新查询,将收到第一个查询。它仅适用于服务器实例的第一次查询。

我从客户端附加代码的相关部分,服务器作为基于select()的服务器工作。

memset(&control, 0, sizeof(control));
control.sun_family = AF_UNIX;
strcpy(control.sun_path, CLIENT_PATH);

bind(sockfd, (struct sockaddr *)&control, sizeof(struct sockaddr_un));

memset(&server, 0, sizeof(server));
server.sun_family = AF_UNIX;
strcpy(server.sun_path, SERVER_PATH);

while(1) {
FD_ZERO(&readfds);
FD_SET(sockfd, &readfds);
FD_SET(STDIN, &readfds);

    if (select(sockfd+1, &readfds, NULL, NULL, NULL) < 0) {
        close(sockfd);
        perror("control: select");
        exit(1);
    }

    if (FD_ISSET(sockfd, &readfds)) {
        if ((nbytes = recvfrom(sockfd, buf, sizeof(buf), 0, NULL, NULL)) < 0) {
            perror("recv");
        }
        buf[nbytes] = '\0';
        printf(">> %s", buf);
    }

    if (FD_ISSET(STDIN, &readfds)) {
        nbytes = read(STDIN, buf, sizeof buf);
        buf[nbytes] = '\0';
        if (sendto(sockfd, buf, strlen(buf), 0,
                (struct sockaddr *)&server, sizeof(struct sockaddr_un)) < strlen(buf)) {
            perror("send");
        }
    }
}

1 个答案:

答案 0 :(得分:0)

请参见手册页:http://linux.die.net/man/2/select。 “如果超时为NULL(无超时),则select()可以无限期地阻塞。”您为它分配了NULL(select()的最后一个参数),然后select()将阻塞,直到套接字准备就绪。

所以你需要做的是为超时分配一个struct timeval。 0秒0 usec超时意味着立即返回,正数意味着当套接字准备好使用(在您的情况下为读取)或超时时返回。