【问题标题】:what can cause select() not to return?什么会导致 select() 不返回?
【发布时间】:2015-04-03 16:09:43
【问题描述】:

我正在编写一个通过 Unix 域套接字进行通信的客户端和服务器(对于 POSIX 开发课程中的一项任务,只是为了学习这个概念)。发生的情况是客户端向服务器发送命令,服务器解析它们并以答案作为响应。我的问题是客户端没有收到对服务器的第一个查询,这与其他客户端不同。

更清楚地说,客户端不会收到对给定服务器实例的第一个查询,如果我关闭客户端实例并打开一个新实例,则会收到第一个查询。它仅发生在对 server 实例的第一次查询中。

我正在从客户端附加代码的相关部分,服务器作为基于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");
        }
    }
}

【问题讨论】:

  • a) recvfrom && sendto == UDP b) UDP && the first query to the server isn't received by the clients, unlike the rest of them. == Not an error。 ...如果您想要可靠的通信,请使用 TCP 或自己实现或使用一些 lib 或...
  • perror() 在 close() 之后被调用 当试图从 select() 显示问题时,这不是一个可靠的顺序,建议颠倒 perror 和 close 语句的顺序
  • 建议在 select() (最后一个参数)上设置一个超时并检查是否过期。发生超时时,循环返回以再次发送字符串。请记住“刷新”超时变量,因为某些操作系统(想到 Linux)会修改该变量
  • @deviantfan,这些是 UDS 数据报。我不明白 b),你能详细说明一下吗?
  • @user3629249,这确实是一个很好的做法,但是因为我知道服务器收到了查询,而且我知道它只发生在第一个查询中 - 我相信再次发送第一个查询会是一个忽略问题本身的丑陋修复。

标签: c sockets posix


【解决方案1】:

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

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

【讨论】:

  • 目前尚不清楚向 select() 添加超时是否可以解决他的根本问题,即未收到预期的查询数据。
  • @JeremyFriesner,我同意。我正在检查数据是否是从服务器发送的,并且确实已发送。
  • 我已将超时设置为 0,但问题仍然存在。我开始认为这可能与我 recvfrom() 然后使用有关发件人的详细信息到 sendto() 查询答案这一事实有关。
猜你喜欢
  • 2016-10-09
  • 2015-06-02
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多