1

我有一个简短的测试程序来获取本地机器的 IP 地址。在 Raspbian 上,仅返回环回地址,而 OS X 上的相同代码返回普通 IP 和环回 IP。

代码是

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <iostream>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <errno.h>
using namespace std;

int main (int argc, char * const argv[]) {
  char hostname[128];

  int result = gethostname(hostname, 127);
  if (result != 0)  {
    cout << "FATAL: gethostname failed: " << result << errno;
    return -1;
  }
  cout << "Host name is " << hostname << endl;

  struct addrinfo hints, *res;
  int errcode;
  char addrstr[100];
  void *ptr;

  memset(&hints, 0, sizeof(hints));
  hints.ai_family = PF_UNSPEC;
  hints.ai_socktype = SOCK_STREAM;
  hints.ai_flags |= AI_CANONNAME;

  errcode = getaddrinfo(hostname, NULL, &hints, &res);
  if (errcode != 0) {
    perror("getaddrinfo");
    return -1;
  }
  while(res) {
    inet_ntop(res->ai_family, res->ai_addr->sa_data, addrstr, 100);
    switch(res->ai_family) {
      case AF_INET:
        ptr = &((struct sockaddr_in *)res->ai_addr)->sin_addr;
        break;
      case AF_INET6:
        ptr = &((struct sockaddr_in6 *)res->ai_addr)->sin6_addr;
        break;
      default:
        cout << "Unknown family" << endl;
    }
    inet_ntop(res->ai_family, ptr, addrstr, 100);
    printf("IPV%d address: %s (%s)\n", res->ai_family == PF_INET6 ? 6 : 4, addrstr, res->ai_canonname);
    res = res->ai_next;
  }
  return 0;
}

在 OS XI 上获得:

[Coyote]collector$ ./quick
Host name is Coyote.local
IPV4 address: 192.168.1.108 (coyote.local)
IPV6 address: fe80::223:dfff:fea0:a230 (coyote.local)

但在 Raspbian 上,我只得到:

pi:~$ ./quick
Host name is pi
IPV4 address: 127.0.1.1 (pi)

然而

pi:~$ ifconfig
eth0      Link encap:Ethernet  HWaddr b8:27:eb:62:15:fc  
          inet addr:192.168.1.162  Bcast:192.168.1.255  Mask:255.255.255.0
[...]

我需要在 Raspbian 上做什么才能获得正确的结果?

4

1 回答 1

1

这可能是由于您的/etc/hosts文件的内容。如果您的目标是获取系统的 IP 地址,请考虑getifaddrs改用。它起源于 BSD,但也适用于 Linux。

如果您想确定在某个地方连接时将使用哪个地址,最好的方法是连接然后使用getsockname. 另一种选择是读取路由表。

此外,getnameinfo通常是比 更好的选择inet_ntop

于 2014-06-17T21:43:44.777 回答