我想获取我的程序启动的计算机的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地址的最佳方法是什么?
当你将套接字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上的客户端。
你可以打电话给ioctl(袜子,SIOCGIFADDR,adr)
见netdevice(7)
根据@Remy Lebeau
的回答,我写了一个返回当前机器地址的函数。我只在macOS High Sierra上测试了这个。
interfaec
可以是lo0
,en0
等之类的任何东西。
ipVersion
可以是AF_INET
或AF_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的初学者,如果有什么不对请告诉我。