我第一次没有获得正确的客户端IP地址。这是我的服务器主代码。
server.c:
int main()
{
int sockfd, connfd, lisfd, retval;
struct sockaddr_in servaddr, clntaddr;
socklen_t client;
struct packet pckt;
int clnt_len;
if((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
perror("socket");
exit(1);
}
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(8000);
retval = bind(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
if(retval) {
perror("bind");
}
if((lisfd = listen(sockfd, 4)) < 0) {
perror("listen");
exit(3);
}
clnt_len = sizeof(clnt_len)-1; // clnt_len = sizeof(clnt_len);
while(1) {
if((connfd = accept(sockfd, (struct sockaddr *)&clntaddr, &clnt_len)) < 0) {
perror("accept");
exit(2);
}
printf("IP address is: %s\n", inet_ntoa(clntaddr.sin_addr));
read(connfd, &pckt, sizeof(pckt));
printf("%s\n", pckt.msg);
}
}
packet
是在hdr.h
中声明的结构,如下所示。struct packet {
char msg[50];
int cmd;
};
输出为:
IP address is: 255.127.0.0
Hi
IP address is: 127.0.0.1
Hello
IP address is: 127.0.0.1
Good
...
第一次,我获得了错误的IP。我不明白这是怎么回事?如何获得正确的IP?
最佳答案
如accept(2)手册页中所述,在调用clnt_len
之前,应将clntaddr
初始化为accept()
中的可用大小:
while(1) {
clnt_len = sizeof clntaddr;
if ((connfd = accept(sockfd, (struct sockaddr *)&clntaddr, &clnt_len)) < 0) {
perror("accept");
exit(2);
}
编辑:
clnt_len = sizeof(clnt_len)-1; // clnt_len = sizeof(clnt_len);
是有问题的行。它将clnt_len
设置为无效值:比套接字长度变量的长度小一,这应该是clntaddr
中可用的内存量。它小于
sizeof (struct sockaddr_in)
,因此,当clntaddr
通过第一个连接返回时,将不会分配accept()
。 clnt_len
确实分配了客户端地址的长度(是的,允许它大于其原始值;这意味着clntaddr
不够大;有关详细信息,请参见man page),因此,在以下连接上,clnt_len
是正确的大小,您将获得第二个及以后的连接的地址,只缺少初始连接地址。要解决此问题,请将
clnt_len = sizeof clntaddr;
行添加到while (1)
lopp主体的开头。有什么问题吗
关于c - 第一次没有正确获取客户端IP,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/26691476/