我正在编写一个非常简单的服务器应用程序,只是为了测试一些代码。 创建套接字并将其连接到本地主机和某个端口后,我想使用
bind()
来了解传入连接何时到达绑定套接字。之后,应用程序应将消息打印到一定长度,然后select()
。我的问题基本上是,当我只期望一个连接时,是否需要使用
exit()
和
listen()
(请记住这只是为了测试)。我相信在这种情况下不需要这些函数,只需要接受多个传入请求。难道我错了?考虑到上述想法,我编写了以下代码
accept()
TCPcreate()
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);
}
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()
不会阻止等待连接。相反,它立即返回 1。如果我取消注释
select()
,我得到的是 printf()
我在这里缺少什么?...
select returns 1
fd = 3
is set? yes
-1
(blank line)
的 POSIX 规范,返回的文件描述符已准备好读取、写入,或者有错误条件。这不会将“
select()
会成功的套接字”列为可检测条件之一。因此,您需要使用
listen()
和
listen()
;只有在接受连接后才能在描述符上使用 accept()
正如 Gonçalo Ribeiro
所指出的,select()
的规范还指出:
如果套接字当前正在侦听,则如果已收到传入连接请求,则应将其标记为可读,并且对
select()
函数的调用应在不阻塞的情况下完成。
这意味着您必须在绑定的套接字上执行
accept()
,但您可以在多个套接字上等待传入连接。
选择的问题在于您的代码中 - 将选择保留在循环中。由于它是非阻塞调用,因此它只会检查是否有人在听。所以,你可以使用循环来多次检查监听。