我正在编写一个非常简单的服务器应用程序,仅用于测试一些代码。
在创建套接字并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)
我在这里缺少什么?...
答案 0 :(得分:1)
如果查看select()
的POSIX规范,则返回的文件描述符已准备好进行读取,写入或出现错误。这不列出'listen()
将成功的套接字作为可检测条件之一。因此,您需要使用listen()
和;只有在您接受连接后,才能在描述符上使用accept()
select()
。
正如Gonçalo Ribeiro所述,select()
的规范也说明了:
如果套接字当前正在侦听,那么如果收到传入的连接请求,则应将其标记为可读,并且对
accept()
函数的调用应完成而不会阻塞。
这意味着您必须在绑定套接字上完成listen()
,但您可以在多个套接字上等待传入连接。
答案 1 :(得分:0)
如果你想要阻止呼叫 - 请使用listen()。
选择的问题在于你的代码是 - 保持选择在循环中。因为它是一个非阻塞呼叫,它只会检查一个人是否在那里听。因此,您可以使用循环检查多次收听。