我正在编写一个非常简单的服务器应用程序,只是为了测试一些代码。
创建套接字并将其绑定(bind)到本地主机和某个端口后,我想使用 select() 来了解传入连接何时到达绑定(bind)套接字。之后,应用程序应打印消息达到一定长度,然后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()
,我得到的是
select returns 1
fd = 3
is set? yes
-1
(blank line)
我在这里缺少什么?...
最佳答案
如果你看看 select()
的 POSIX 规范,返回的文件描述符已准备好读取、写入,或者有错误条件。这不会列出“listen()
上的套接字”将成功”作为可检测条件之一。因此,您需要使用 listen()
和 ;只有在接受连接后才能在描述符上使用 accept()
select()
。
如Gonçalo Ribeiro注意,select()
的规范还指出:
If the socket is currently listening, then it shall be marked as readable if an incoming connection request has been received, and a call to the
accept()
function shall complete without blocking.
这意味着您必须在绑定(bind)的套接字上执行listen()
,但您可以在多个套接字上等待传入连接。
关于c - select() 返回时没有传入连接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22130276/