是什么导致 select() 不是 return?
what can cause select() not to return?
我正在编写一个通过 Unix 域套接字进行通信的客户端和服务器(用于 POSIX 开发 class 中的一项任务,只是为了学习概念)。发生的事情是客户端向服务器发送命令,服务器解析它们并响应一个答案。我的问题是客户端没有收到对服务器的第一个查询,这与其他客户端不同。
更清楚地说,客户端不会收到对给定服务器实例的第一个查询,如果我关闭客户端实例并打开一个新实例,则会收到第一个查询。它仅发生在对 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] = '[=12=]';
printf(">> %s", buf);
}
if (FD_ISSET(STDIN, &readfds)) {
nbytes = read(STDIN, buf, sizeof buf);
buf[nbytes] = '[=12=]';
if (sendto(sockfd, buf, strlen(buf), 0,
(struct sockaddr *)&server, sizeof(struct sockaddr_un)) < strlen(buf)) {
perror("send");
}
}
}
参见手册页:http://linux.die.net/man/2/select。
"If timeout is NULL (no timeout), select() can block indefinitely." 您将其分配为 NULL(select() 的最后一个参数),然后 select() 将阻塞直到套接字准备就绪。
所以你需要做的是为超时分配一个struct timeval。 0 sec 0 usec 超时意味着 return 立即,正数意味着 return 当套接字准备好使用(在您的情况下为读取)或超时时。
我正在编写一个通过 Unix 域套接字进行通信的客户端和服务器(用于 POSIX 开发 class 中的一项任务,只是为了学习概念)。发生的事情是客户端向服务器发送命令,服务器解析它们并响应一个答案。我的问题是客户端没有收到对服务器的第一个查询,这与其他客户端不同。
更清楚地说,客户端不会收到对给定服务器实例的第一个查询,如果我关闭客户端实例并打开一个新实例,则会收到第一个查询。它仅发生在对 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] = '[=12=]';
printf(">> %s", buf);
}
if (FD_ISSET(STDIN, &readfds)) {
nbytes = read(STDIN, buf, sizeof buf);
buf[nbytes] = '[=12=]';
if (sendto(sockfd, buf, strlen(buf), 0,
(struct sockaddr *)&server, sizeof(struct sockaddr_un)) < strlen(buf)) {
perror("send");
}
}
}
参见手册页:http://linux.die.net/man/2/select。 "If timeout is NULL (no timeout), select() can block indefinitely." 您将其分配为 NULL(select() 的最后一个参数),然后 select() 将阻塞直到套接字准备就绪。
所以你需要做的是为超时分配一个struct timeval。 0 sec 0 usec 超时意味着 return 立即,正数意味着 return 当套接字准备好使用(在您的情况下为读取)或超时时。