10.获取网卡信息
2017-09-04 18:03
176 查看
我们可以使用ioctl函数获取网卡信息
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <net/if.h>
#include <stdio.h>
void get_if_info()
{
int sockfd, len, flags;
struct ifconf ifc;
char *buf, *ptr, *cptr, lastname[IFNAMSIZ];
char ipaddr[INET_ADDRSTRLEN], ip6addr[INET6_ADDRSTRLEN];
struct ifreq * ifr;
struct sockaddr_in * sa4;
struct sockaddr_in6 * sa6;
sockfd = socket(AF_INET, SOCK_STREAM, 0);
// 创建用于存放网卡列表的缓冲区
// 这里创建了可以存放100个ifreq结构的缓冲区
// 实际场景下为了避免缓冲区大小不足, 导致结果被截断,
// 我们可以采取试探性的策略进行动态增长
len = 100 * sizeof(struct ifreq);
buf = malloc(len);
ifc.ifc_len = len;
ifc.ifc_ifcu.ifcu_buf = buf;
// ioctl函数:SIOCGIFCONF, 获取所有接口的列表
// 这里的ifc是个值-结果参数, 在调用过程中, 内核会修改里面的值, 譬如ifc_len
ioctl(sockfd, SIOCGIFCONF, &ifc);
// 对返回的接口列表进行遍历
for(ptr = buf; ptr < buf + ifc.ifc_len; ptr += sizeof(struct ifreq))
{
ifr = (struct ifreq *)ptr;
// 我们对同一个接口只处理一次, 拥有多个IP的接口也只处理一次, 用lastname存放上一个被处理的接口名字
// 我们可以给一个网卡配置多个IP(别名地址), Linux下用于别名地址的接口名字中含有一个冒号(例如eth0:1)
if((cptr = strchr(ifr->ifr_name, ':')) != NULL)
*cptr = '\0';
if(strncmp(lastname, ifr->ifr_name, IFNAMSIZ) == 0)
continue;
// 获取网卡的名字, 并记录到lastname
printf("iface: %s\n", ifr->ifr_name);
memcpy(lastname, ifr->ifr_name, IFNAMSIZ);
// 获取网卡的IP地址
switch(ifr->ifr_ifru.ifru_addr.sa_family)
{
case AF_INET6:
sa6 = (struct sockaddr_in6 *)(&ifr->ifr_ifru.ifru_addr);
inet_ntop(AF_INET6, &sa6->sin6_addr, ip6addr, sizeof(ip6addr));
printf("\tinet6 addr: %s\n", ip6addr);
break;
case AF_INET:
sa4 = (struct sockaddr_in *)(&ifr->ifr_ifru.ifru_addr);
inet_ntop(AF_INET, &sa4->sin_addr, ipaddr, sizeof(ipaddr));
printf("\tinet addr: %s\n", ipaddr);
break;
default:
printf("\tip addr unknown\n");
break;
}
// 获取网卡的标志, 并判断网卡是否处于工作状态
ioctl(sockfd, SIOCGIFFLAGS, ifr);
flags = ifr->ifr_flags;
if(flags & IFF_UP)
printf("\tstatus: UP\n");
else
printf("\tstatus: DOWN\n");
// 获取网卡的MTU(最大传输单元)
ioctl(sockfd, SIOCGIFMTU, ifr);
printf("\tmtu: %d\n", ifr->ifr_mtu);
}
// 释放缓冲区
free(buf);
}
int main()
{
get_if_info();
}
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <net/if.h>
#include <stdio.h>
void get_if_info()
{
int sockfd, len, flags;
struct ifconf ifc;
char *buf, *ptr, *cptr, lastname[IFNAMSIZ];
char ipaddr[INET_ADDRSTRLEN], ip6addr[INET6_ADDRSTRLEN];
struct ifreq * ifr;
struct sockaddr_in * sa4;
struct sockaddr_in6 * sa6;
sockfd = socket(AF_INET, SOCK_STREAM, 0);
// 创建用于存放网卡列表的缓冲区
// 这里创建了可以存放100个ifreq结构的缓冲区
// 实际场景下为了避免缓冲区大小不足, 导致结果被截断,
// 我们可以采取试探性的策略进行动态增长
len = 100 * sizeof(struct ifreq);
buf = malloc(len);
ifc.ifc_len = len;
ifc.ifc_ifcu.ifcu_buf = buf;
// ioctl函数:SIOCGIFCONF, 获取所有接口的列表
// 这里的ifc是个值-结果参数, 在调用过程中, 内核会修改里面的值, 譬如ifc_len
ioctl(sockfd, SIOCGIFCONF, &ifc);
// 对返回的接口列表进行遍历
for(ptr = buf; ptr < buf + ifc.ifc_len; ptr += sizeof(struct ifreq))
{
ifr = (struct ifreq *)ptr;
// 我们对同一个接口只处理一次, 拥有多个IP的接口也只处理一次, 用lastname存放上一个被处理的接口名字
// 我们可以给一个网卡配置多个IP(别名地址), Linux下用于别名地址的接口名字中含有一个冒号(例如eth0:1)
if((cptr = strchr(ifr->ifr_name, ':')) != NULL)
*cptr = '\0';
if(strncmp(lastname, ifr->ifr_name, IFNAMSIZ) == 0)
continue;
// 获取网卡的名字, 并记录到lastname
printf("iface: %s\n", ifr->ifr_name);
memcpy(lastname, ifr->ifr_name, IFNAMSIZ);
// 获取网卡的IP地址
switch(ifr->ifr_ifru.ifru_addr.sa_family)
{
case AF_INET6:
sa6 = (struct sockaddr_in6 *)(&ifr->ifr_ifru.ifru_addr);
inet_ntop(AF_INET6, &sa6->sin6_addr, ip6addr, sizeof(ip6addr));
printf("\tinet6 addr: %s\n", ip6addr);
break;
case AF_INET:
sa4 = (struct sockaddr_in *)(&ifr->ifr_ifru.ifru_addr);
inet_ntop(AF_INET, &sa4->sin_addr, ipaddr, sizeof(ipaddr));
printf("\tinet addr: %s\n", ipaddr);
break;
default:
printf("\tip addr unknown\n");
break;
}
// 获取网卡的标志, 并判断网卡是否处于工作状态
ioctl(sockfd, SIOCGIFFLAGS, ifr);
flags = ifr->ifr_flags;
if(flags & IFF_UP)
printf("\tstatus: UP\n");
else
printf("\tstatus: DOWN\n");
// 获取网卡的MTU(最大传输单元)
ioctl(sockfd, SIOCGIFMTU, ifr);
printf("\tmtu: %d\n", ifr->ifr_mtu);
}
// 释放缓冲区
free(buf);
}
int main()
{
get_if_info();
}
相关文章推荐
- python通过snmp协议运用多线程获取多台主机网卡信息,写入数据库
- C++ 中通过GetAdaptersInfo获取网卡配置和Ip地址信息
- linux c获取系统网卡信息
- VC 获取网卡信息
- 关于获取网卡信息的一点消息
- VB6获取网卡信息
- 获取网卡信息
- C++ 中通过GetAdaptersInfo获取网卡配置和Ip地址信息
- 获取外网IP和本机所有网卡地址信息
- 获取网卡信息
- linux下的shell命令的编写,以及java怎样调用linux的shell命令(java怎样获取linux上的网卡的ip信息)
- Qt5获取网卡/IP等信息
- Linux上获取所有网卡信息
- 用加载动态库Iphlpapi.dll的方法获取网卡信息
- 网络公共函数(大小端字节转换、网卡/ip信息获取等。)
- 一个专门获取网卡信息的 没有Iphlpapi.h与Iphlpapi.lib,怎么处理?
- 获取适配器网卡信息
- C++ 中通过GetAdaptersInfo获取网卡配置和Ip地址信息
- java 获取本机硬件信息 cpu 内存 硬盘 网卡等
- [原blog文章]获取网卡信息(一)