C语言 我怎样才能知道 C 中接口的 IP 地址?
声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow
原文地址: http://stackoverflow.com/questions/4139405/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me):
StackOverFlow
How can I get to know the IP address for interfaces in C?
提问by gvalero87
Let's say I'm running a program called IpAddresses.c. I want that program to get all IP addresses this device has according to each interface. Just like ifconfig. How can I do that?
假设我正在运行一个名为 IpAddresses.c 的程序。我希望该程序根据每个接口获取该设备的所有 IP 地址。就像ifconfig一样。我怎样才能做到这一点?
I don't know much about ioctl, but I read it might help me.
我对ioctl了解不多,但我读过它可能对我有帮助。
回答by chrisaycock
Just use getifaddrs(). Here's an example:
只需使用getifaddrs()。下面是一个例子:
#include <arpa/inet.h>
#include <sys/socket.h>
#include <ifaddrs.h>
#include <stdio.h>
int main ()
{
struct ifaddrs *ifap, *ifa;
struct sockaddr_in *sa;
char *addr;
getifaddrs (&ifap);
for (ifa = ifap; ifa; ifa = ifa->ifa_next) {
if (ifa->ifa_addr && ifa->ifa_addr->sa_family==AF_INET) {
sa = (struct sockaddr_in *) ifa->ifa_addr;
addr = inet_ntoa(sa->sin_addr);
printf("Interface: %s\tAddress: %s\n", ifa->ifa_name, addr);
}
}
freeifaddrs(ifap);
return 0;
}
And here's the output I get on my machine:
这是我在我的机器上得到的输出:
Interface: lo Address: 127.0.0.1
Interface: eth0 Address: 69.72.234.7
Interface: eth0:1 Address: 10.207.9.3
回答by jschmier
Here's some Linuxsample code that might help you out.
这是一些可能对您有所帮助的Linux示例代码。
#include <stdio.h>
#include <net/if.h>
#include <netinet/in.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/socket.h>
#define INT_TO_ADDR(_addr) \
(_addr & 0xFF), \
(_addr >> 8 & 0xFF), \
(_addr >> 16 & 0xFF), \
(_addr >> 24 & 0xFF)
int main()
{
struct ifconf ifc;
struct ifreq ifr[10];
int sd, ifc_num, addr, bcast, mask, network, i;
/* Create a socket so we can use ioctl on the file
* descriptor to retrieve the interface info.
*/
sd = socket(PF_INET, SOCK_DGRAM, 0);
if (sd > 0)
{
ifc.ifc_len = sizeof(ifr);
ifc.ifc_ifcu.ifcu_buf = (caddr_t)ifr;
if (ioctl(sd, SIOCGIFCONF, &ifc) == 0)
{
ifc_num = ifc.ifc_len / sizeof(struct ifreq);
printf("%d interfaces found\n", ifc_num);
for (i = 0; i < ifc_num; ++i)
{
if (ifr[i].ifr_addr.sa_family != AF_INET)
{
continue;
}
/* display the interface name */
printf("%d) interface: %s\n", i+1, ifr[i].ifr_name);
/* Retrieve the IP address, broadcast address, and subnet mask. */
if (ioctl(sd, SIOCGIFADDR, &ifr[i]) == 0)
{
addr = ((struct sockaddr_in *)(&ifr[i].ifr_addr))->sin_addr.s_addr;
printf("%d) address: %d.%d.%d.%d\n", i+1, INT_TO_ADDR(addr));
}
if (ioctl(sd, SIOCGIFBRDADDR, &ifr[i]) == 0)
{
bcast = ((struct sockaddr_in *)(&ifr[i].ifr_broadaddr))->sin_addr.s_addr;
printf("%d) broadcast: %d.%d.%d.%d\n", i+1, INT_TO_ADDR(bcast));
}
if (ioctl(sd, SIOCGIFNETMASK, &ifr[i]) == 0)
{
mask = ((struct sockaddr_in *)(&ifr[i].ifr_netmask))->sin_addr.s_addr;
printf("%d) netmask: %d.%d.%d.%d\n", i+1, INT_TO_ADDR(mask));
}
/* Compute the current network value from the address and netmask. */
network = addr & mask;
printf("%d) network: %d.%d.%d.%d\n", i+1, INT_TO_ADDR(network));
}
}
close(sd);
}
return 0;
}
回答by Luiz Normando
The solution using getifaddrs()is great. I would suggest only one improve:
使用getifaddrs()的解决方案很棒。我只建议改进一项:
--- chrisaycock
+++ normando
@@ -11,7 +11,7 @@
getifaddrs (&ifap);
for (ifa = ifap; ifa; ifa = ifa->ifa_next) {
- if (ifa->ifa_addr->sa_family==AF_INET) {
+ if (ifa->ifa_addr && ifa->ifa_addr->sa_family==AF_INET) {
sa = (struct sockaddr_in *) ifa->ifa_addr;
addr = inet_ntoa(sa->sin_addr);
printf("Interface: %s\tAddress: %s\n", ifa->ifa_name, addr);
Just because I myself got a Segmentation Fault.
只是因为我自己遇到了分段错误。
回答by ninjalj
See this other Stack Overflow question, Enumerating each IP address assigned to network interfaces.
请参阅其他堆栈溢出问题,枚举分配给网络接口的每个 IP 地址。
In summary, you can use:
总之,您可以使用:
ioctl(SIOCGIFCONF)-> the traditional ioctlgetifaddrs()-> from BSDi, now also on Linux and the BSD's.- RTNETLINK (Linux)
ioctl(SIOCGIFCONF)-> 传统的ioctlgetifaddrs()-> 来自 BSDi,现在也在 Linux 和 BSD 上。- RTNETLINK (Linux)
回答by macgarden
You could try something like that:
你可以尝试这样的事情:
struct ifreq ifr[MAX_INTERFACES];
struct ifconf ifc;
memset(ifr, 0, sizeof(ifr));
ifc.ifc_len = sizeof(ifr);
ifc.ifc_req = ifr;
// Get the list of interfaces
if (ioctl(sock, SIOCGIFCONF, &ifc) == -1) {
fprintf(stderr, "ioctl SIOCGIFCONF failed: %d", errno);
}
struct ifreq *ifr_iterator = ifc.ireq;
int i = 0;
size_t len;
while (i < ifc.ifc_len) {
/* DO STUFF */
// Maybe some more filtering based on SIOCGIFFLAGS
// Your code
// Use ifr_iterator-> ...
len = IFNAMSIZ + ifr_iterator->ifr_addr.sa_len;
ifr_iterator = (struct ifreq *)((char *)ifr_iterator + len);
i += len;
}
回答by Steve Townsend
Check out the (Windows specific) IP Helper API- fortunately you don't need ioctlfor this on Windows.
查看(特定于Windows 的)IP Helper API- 幸运的是,您ioctl在 Windows 上不需要这个。

