有时我们需要动态的根据网卡编号来获取IP地址,如果在配置文件里配置ip的话就不能动态的实现了,而linux下的ioctl()函数则提供了实现的途径。
ioctl是设备驱动程序中对设备的I/O通道进行管理的函数。所谓对I/O通道进行管理,就是对设备的一些特性进行控制,例如串口的传输波特率、马达的转速等等。它的调用个数如下:
int ioctl(int fd, ind cmd, …);
其中fd是用户程序打开设备时使用open函数返回的文件标示符,cmd是用户程序对设备的控制命令,至于后面的省略号,那是一些补充参数,一般最多一个,这个参数的有无和cmd的意义相关。
ioctl函数是文件结构中的一个属性分量,就是说如果你的驱动程序提供了对ioctl的支持,用户就可以在用户程序中使用ioctl函数来控制设备的I/O通道。
具体代码如下:
#include<errno.h>
#include<sys/type.h>
#include <unistd.h>
#inlcude <netinet/tcp.h>
#include<sys/socket.h>
#include<sys/ioctl.h>
#include<arpa/inet.h>
#include<net/if.h>
#include<cstdlib>
#include<iostream>
/****************
brief:根据网卡的名称编号获取指定的IP地址
param:ifName,需要取IP地址的网卡名称
return 获取的指定网卡上的IP地址
****************/
const char* getIPByIfName(const char *ifName)
{
int sock;
struct ifreq ifr; //结构体头文件#include<net/if.h>
static const char *none_if="0.0.0.0";
if(Null == ifName)
return none_ip;
sock = ::socket(AF_INET,SOCK_DGRAM,0);
if(-1 == sock)
return none_ip;
bzero(ifr.ifr_name,sizeof(ifr.ifr_name));
strncpy(ifr.ifr_name,ifName,sizeof(ifr.ifr_name)-1);
if(-1 == ioctl(sock,SIOCGIFADDR,&ifr))//获取网卡的IP地址
{
TEMP_FAILURE_ENTRY(::close(sock));
return none_ip;
}
TEMP_FAILURE_ENTRY(::close(sock));
return inet_ntoa(((struct sockaddr_in*)&ifr.ifr_addr)->sin_addr);//将网络地址转换成字符串
}
ioctl()的详细解释可以参考:https://www.cnblogs.com/tdyizhen1314/p/4896689.html