我开发了(TCP)服务器来监听客户端并与之交互。现在我正在尝试调整该服务器代码来监听多个客户端。我想使用select,但我对我发现的一些例子和解释感到困惑。
我一直在阅读:http://support.sas.com/documentation/onlinedoc/sasc/doc750/html/lr2/select.htm和http://support.sas.com/documentation/onlinedoc/sasc/doc750/html/lr2/select.htm这两个都被推荐给我..:S
第一个网站的例子似乎不太复杂..(虽然仍然不知道如何使其适应我的代码,因为我非常非常非常新的网络等)但我担心我错过了关键方面由于第二个网站的例子的复杂性。
当只监听一个客户端时,下面是我的服务器代码的快照(包括一些伪代码以最小化不那么重要的东西):
int main(int argc, char *argv[])
{
int sockfd, newsockfd, portno, clilen;
char buffer[3];
struct sockaddr_in serv_addr, cli_addr;
int n;
if (argc < 2)
{
fprintf(stderr,"ERROR, no port provided\n");
exit(1);
}
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0){error("ERROR opening socket");}
bzero((char *) &serv_addr, sizeof(serv_addr));
portno = atoi(argv[1]);
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(portno);
if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0)
{error("ERROR on binding");}
listen(sockfd,5);
clilen = sizeof(cli_addr);
newsockfd = accept(sockfd, (struct sockaddr *) &cli_addr, &clilen);
if (newsockfd < 0){error("ERROR on accept");}
while (UNTIL END OF FILE)
{
<DO SOME FILE READING STUFF>
n = write(newsockfd, "test/n", 5);
if (n < 0){error("ERROR writing to socket");}
bzero(buffer,3);
n = read(newsockfd,buffer,3);
if (n < 0){error("ERROR reading from socket");}
buffer[n] = 0;
<DO SOME STRING STUFF>
while(done != 1)
{
bzero(buffer,3);
n = read(newsockfd,buffer,3);
if (n < 0){error("ERROR reading from socket");}
buffer[n] = 0;
if(strcmp(buffer, "CO"))
{
done = 1;
}
}
done = 0;
}
<DO STUFF>
n = write(newsockfd, "DN\n", 2);
if (n < 0){error("ERROR writing to socket");}
close(sockfd);
close(newsockfd);
return 0;
}
哪个网站的select()示例最适合我正在尝试做的事情(更改服务器代码以收听多个客户端)?有人可以用一些更简单的术语解释select()吗? (因为我对这个和所有人都很新......)
谢谢!
答案 0 :(得分:7)
您需要围绕accept()
和以下代码包装循环,以便您可以在程序的整个生命周期内接受多个连接。然后,您需要确定服务器将如何处理多个连接。你有几个选择:
select()
来确定哪些文件描述符已准备好。在前两种情况下,子线程或进程是一心一意处理一个客户端;它将在客户端正在研究如何响应它发送的内容时等待,这意味着其他线程或进程在CPU处转向。主题上有一些变化,它们预先发布了一些工作线程或进程,并安排那些工作负载,但这更复杂。
在最后一种情况下,您将设置一个您希望select()
处理的文件描述符数组,然后将该列表的副本传递给select()
(因为它遍及全局它)。当你有工作要做(因为select()
返回)时,如果已经准备就在你正在监听的文件描述符上执行accept()
,或者你从已经打开的套接字描述符中读取他们准备好了。