如何获取已接受的入站套接字的IP地址?

3
我的问题是:
服务器将创建一个套接字,绑定到给定端口和地址= INADDR_ANY。
监听() & 接受() 新连接。然后,我们可以从accept()中获取客户端的IP地址。
现在,我想知道服务器的IP地址,因为服务器的主机上有多个NIC。
如何知道已接受的入站套接字所使用的网络接口的IP地址?
我尝试了getsockname,它给了我端口号,但IP是全零。
更新:以下是代码:
Server.c(头文件已删除)
int main(void)
{

    struct sockaddr_in stSockAddr;
    int res, addr_len, SocketFD, ConnectFD;

    struct sockaddr_in addr;

    SocketFD = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);

    if(-1 == SocketFD)
    {
      perror("can not create socket");
      //exit(EXIT_FAILURE);
      return -1;
    }

    memset(&stSockAddr, 0, sizeof stSockAddr);
    stSockAddr.sin_family = AF_INET;
    stSockAddr.sin_port = htons(49335);
    stSockAddr.sin_addr.s_addr = INADDR_ANY;

    if(-1 == bind(SocketFD,(struct sockaddr *)&stSockAddr, sizeof stSockAddr))
    {
      perror("error bind failed");
      close(SocketFD);
     return -1;
    }


    printf("going to listen!\n"); 
    if(-1 == listen(SocketFD, 10))
    {   
      perror("error listen failed");
      close(SocketFD);
      //exit(EXIT_FAILURE);
      return -1; 
    }   

    ConnectFD = accept(SocketFD, NULL, NULL);

        if(0 > ConnectFD)
        {   
                perror("error accept failed");
                close(SocketFD);
                //exit(EXIT_FAILURE);
                return -1; 
        }   

    addr.sin_family = AF_INET;
    res = getsockname (ConnectFD, (struct sockaddr *)&addr, &addr_len);
    // if you remove the following comment, that means, if you call 
    // two times of getsockname, the result will be correct.
    //res = getsockname (ConnectFD, (struct sockaddr *)&addr, &addr_len);

    printf("addr:%x\n", addr.sin_addr.s_addr);

    while(1) {
        if (getchar() == 'q')
                        break;
    }

    close(ConnectFD);
    close(SocketFD);
    return 0;
}

以下是client.c文件:
int main(void)
{

    struct sockaddr_in stSockAddr;

    int Res;

    int SocketFD = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);

    if (-1 == SocketFD)
    {
      perror("cannot create socket");
      exit(EXIT_FAILURE);
    }

    memset(&stSockAddr, 0, sizeof stSockAddr);

    stSockAddr.sin_family = AF_INET;
    stSockAddr.sin_port = htons(49335);
    Res = inet_pton(AF_INET, "192.168.1.102", &stSockAddr.sin_addr);

    if (0 > Res)
    {
      perror("error: first parameter is not a valid address family");
      close(SocketFD);
      exit(EXIT_FAILURE);
    }
    else if (0 == Res)
    {
      perror("char string (second parameter does not contain valid ipaddress");
      close(SocketFD);
      exit(EXIT_FAILURE);
    }

    if (-1 == connect(SocketFD, (struct sockaddr *)&stSockAddr, sizeof stSockAddr))
    {
      perror("connect failed");
      close(SocketFD);
      exit(EXIT_FAILURE);
    }

    /* perform read write operations ... */
    printf("client sockfd is successful\n"); 

      while(1) {

          if (getchar() == 'q')
              break;

      }


    shutdown(SocketFD, SHUT_RDWR);

    close(SocketFD);
    return 0;
}

请简单编辑您的问题以提供更多信息。答案仅保留用于解决您的问题的解决方案。您还可以在您的问题下使用评论功能或任何答案来讨论已发布的解决方案或提供澄清。 - Tim Post
3个回答

11

在从bind(2)返回的套接字上不要使用getsockname(2),而是应该在从accept(2)返回的套接字上使用。


8

getsockname()函数可以获取传递的套接字的名称。在本例中,您需要传递由服务器的accept函数创建的套接字。这个套接字在服务器端,所以它的名称和地址与服务器端相关。

如果您想知道"谁连接到了我",您必须使用getpeername()而不是getsockname。

祝好运!


1

如果你删除以下注释,那么当你调用两次getsockname时,结果将是正确的。 res = getsockname (ConnectFD, (struct sockaddr *)&addr, &addr_len);

你必须初始化addr_len。

    addr_len = sizeof(addr);

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接