如何使用套接字地址获取自己的IP地址?

问题描述 投票:7回答:4

我想获取我的程序启动的计算机的IP地址,然后能够将其发送到客户端,但我总是得到0.0.0.1而不是真正的IP地址(例如127.0.0.1)。

我目前能够获得端口,但不能获得IP地址。

我怎么才能得到它?

最好的解决方案是能够用sockaddr_in获得它。这是我目前正在做的事情:

int open_connection(char* ip, int* port)
{
  int                   sock;
  struct sockaddr_in    sin;
  socklen_t             len;
  int                   i;

  i = 0;
  len = sizeof(sin);
  if ((sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
    return (-1);
  bzero(&sin, sizeof(struct sockaddr_in));
  sin.sin_family = AF_INET;
  if (bind(sock, (struct sockaddr *) &sin, sizeof(sin)) != 0)
    perror("Error on bind");
  if (getsockname(sock, (struct sockaddr *)&sin, &len) != 0)
    perror("Error on getsockname");
  strcpy(ip, inet_ntoa(sin.sin_addr)); // IP = 0.0.0.0
  *port = sin.sin_port;
  return (sock);
}

编辑:我理解我用我的思维方式走错了方向。所以我的问题是:获取自己的IP地址的最佳方法是什么?

c linux sockets networking
4个回答
11
投票

当你将套接字bind()到0.0.0.0时,这是套接字在调用getsockname()时唯一可用的IP。这意味着套接字绑定到所有本地接口。为了从套接字获取特定IP,必须将其绑定到特定IP。

无论如何,使用套接字API来获取计算机的本地IP是错误的方法。一个常见的错误是使用gethostname()gethostbyname()getaddrinfo()获取本地IP列表。通常这有效,但它有一些隐藏的陷阱,可以导致虚假的信息,但人们往往忽略这一事实,或者甚至不知道它一开始(我多年不知道它,但后来我学得更好)。

相反,您确实应该使用特定于平台的API来枚举本地网络接口。这将提供更可靠的信息。 Windows有GetAdaptersInfo()GetAdaptersAddresses()。其他平台有getifaddrs()。这些将告诉您哪些本地IP可用。然后,您可以将套接字bind()设置为0.0.0.0以接受任何这些IP上的客户端,或者bind()接收到特定IP以仅接受该IP上的客户端。


4
投票

套接字API允许您枚举分配给您的网络接口的IP地址,但如果您从路由器后面连接到Internet,它不会告诉您“真正的IP”是什么。

了解它的唯一方法是向外面的人询问。这就像FileZilla FTP Server这样的服务器是如何做到的。它们指示您在服务器的设置中将URL配置为“ip.php”脚本,如this one,以便它可以向Internet询问其公共IP地址,以便在被动模式下使用。

您还可以考虑使用STUN,这是一种广泛用于VoIP的协议,用于发现公共IP。


1
投票

你可以打电话给ioctl(袜子,SIOCGIFADDR,adr)

见netdevice(7)


0
投票

根据@Remy Lebeau的回答,我写了一个返回当前机器地址的函数。我只在macOS High Sierra上测试了这个。

interfaec可以是lo0en0等之类的任何东西。

ipVersion可以是AF_INETAF_INET6

long int getInternalAddress(char* interface, sa_family_t ipVersion)
{
  struct ifaddrs *ifaddrHead, *ifaddr;
  /* int_8 */
  sa_family_t family;
  int n;
  char *interfaceName;

  if (getifaddrs(&ifaddrHead) != 0)
  {
    fprintf(stderr, "ifaddrs error");
  }

  /* iterate through address list */
  for (ifaddr = ifaddrHead, n = 0; ifaddr != NULL; ifaddr = ifaddr->ifa_next, n++)
  {
    family = ifaddr->ifa_addr->sa_family;
    interfaceName = ifaddr->ifa_name;

    if (!family || family != ipVersion || strcmp(interfaceName, interface)) continue;

    struct sockaddr *addr = ifaddr->ifa_addr;
    struct sockaddr_in* addr_in = (struct sockaddr_in*) addr;
    long int address = addr_in->sin_addr.s_addr;

    freeifaddrs(ifaddrHead);

    return address;
  }

  freeifaddrs(ifaddrHead);

  return 0;
}

要使用它,

int main()
{
  long int address = getInternalAddress((char*) &"en0", AF_INET);
  printf("%li\n", address);

  return 0;
}

我还是C的初学者,如果有什么不对请告诉我。

© www.soinside.com 2019 - 2024. All rights reserved.