获取本机所有网卡的网卡名、网卡描述、网卡MAC地址、网卡IP、网卡类型等信息及网线是否插入状态

使用windows sdk提供的API函数GetAdaptersInfo()可以获得本机所有网卡的网卡名、网卡描述、网卡MAC地址、网卡IP、网卡类型等信息,并用IP_ADAPTER_INFO结构体存储,使用GetIfEntry()获取网卡的状态,可有效判断出网卡通讯是否正常,是否插入网线等状态......

关于本文更为详细的说明及代码示例详见:http://download.csdn.net/detail/lusirking/9562517

判断网线是否插入

可利用IPHELPAPI中的GetIfEntry()进行获取,代码如下:

bool getAdapterState(DWORD index)
{
MIB_IFROW Info ;    // 存放获取到的Adapter参数
memset(&Info ,0 ,sizeof(MIB_IFROW)) ; 
Info.dwIndex = index ; // dwIndex是需要获取的Adapter的索引,可以通过GetAdaptersInfo和其他相关函数获取
if(GetIfEntry(&Info) != NOERROR)
{
printf("ErrorCode = %d\n" ,GetLastError());
return false;
}
if(Info.dwOperStatus == IF_OPER_STATUS_NON_OPERATIONAL

                || Info.dwOperStatus == IF_OPER_STATUS_UNREACHABLE
                || Info.dwOperStatus == IF_OPER_STATUS_DISCONNECTED 

                || Info.dwOperStatus == IF_OPER_STATUS_CONNECTING)
return false;
else if(Info.dwOperStatus == IF_OPER_STATUS_OPERATIONAL

                  || Info.dwOperStatus == IF_OPER_STATUS_CONNECTED)
return true;
}

MIB_IFROW  介绍:

MIB_IFROW  中有一项 dwOperStatus的参数,它表明当前接口的操作状态

其值如下:

1)IF_OPER_STATUS_NON_OPERATIONAL

LANadapter has been disabled, for example because of an address conflict.

局域网适配器禁用,例如地址冲突

2)IF_OPER_STATUS_UNREACHABLE

WANadapter that is not connected.

WAN适配器未连接

3)IF_OPER_STATUS_DISCONNECTED

ForLAN adapters: network cable disconnected. For WAN adapters: no carrier

局域网适配器:网线未插入。WAN适配器:无信号

4)IF_OPER_STATUS_CONNECTING

WANadapter that is in the process of connecting.

正在处理连接

5)IF_OPER_STATUS_CONNECTED

WANadapter that is connected to a remote peer.

 已连接远端

6)IF_OPER_STATUS_OPERATIONAL

Defaultstatus for LAN adapters 默认状态

经测试

拔下网线时,dwOperStatus值为IF_OPER_STATUS_NON_OPERATIONAL

连接网线时,dwOperStatus 值为IF_OPER_STATUS_OPERATIONAL


IP_ADAPTER_INFO结构体描述如下:

typedef struct _IP_ADAPTER_INFO

{

  struct _IP_ADAPTER_INFO* Next;//指向链表中下一个适配器信息的指针

  DWORD ComboIndex;//预留值

  char AdapterName[MAX_ADAPTER_NAME_LENGTH + 4];//使用ANSI字符串表示的适配器名称

  char Description[MAX_ADAPTER_DESCRIPTION_LENGTH +4];//使用ANSI字符串表示的适配器描述

  UINT AddressLength;//适配器硬件地址以字节计算的长度

  BYTE Address[MAX_ADAPTER_ADDRESS_LENGTH];//硬件地址以BYTE数组所表示

  DWORD Index;//适配器索引

    UINT Type;//适配器类型,主要有以下几种:

    /*

     *   MIB_IF_TYPE_OTHER     1

     *   MIB_IF_TYPE_ETHERNET     6

     *   MIB_IF_TYPE_TOKENRING     9

     *   MIB_IF_TYPE_FDDI     15

     *   MIB_IF_TYPE_PPP     23

     *   MIB_IF_TYPE_LOOPBACK      24

     *   MIB_IF_TYPE_SLIP      28

     */

  UINT DhcpEnabled;//指定这个适配器是否开启DHCP

  PIP_ADDR_STRING CurrentIpAddress;//预留值

  IP_ADDR_STRING IpAddressList;//该适配器的IPv4地址链表

  IP_ADDR_STRING GatewayList;//该适配器的网关IPv4地址链表

  IP_ADDR_STRING DhcpServer;//该适配器的DHCP服务器的IPv4 地址链表

  BOOL HaveWins;

  IP_ADDR_STRING PrimaryWinsServer;

  IP_ADDR_STRING SecondaryWinsServer;

  time_t LeaseObtained;

  time_t LeaseExpires;

} IP_ADAPTER_INFO,*PIP_ADAPTER_INFO;

由于可能有多个网卡,因此struct _IP_ADAPTER_INFO* Next字段为一个链表结构指针,由于一个网卡可能有多个IP,因此IP_ADDR_STRING字段应该也是一个链表结构,其信息如下所示:

typedef struct_IP_ADDR_STRING

{

    struct _IP_ADDR_STRING* Next;  //指向同类型节点,即下一个IP(如果有多IP的话)

    IP_ADDRESS_STRING IpAddress;  //IP地址信息

    IP_MASK_STRING IpMask; //IP子网掩码

    DWORD Context;// 网络表入口。这个值对应着AddIPAddredd和DeleteIPAddress函数中的NTEContext参数

} IP_ADDR_STRING;

IP_ADDR_STRING结构也是一个链表节点

综上所述,用下图来描述网卡的结构存储信息,也许更明朗:


获取网卡信息的代码如下:

#include <WinSock2.h>
#include <Iphlpapi.h>
#include <iostream>
using namespace std;
#pragma comment(lib,"Iphlpapi.lib") //需要添加Iphlpapi.lib库
int __cdecl main()
{
//PIP_ADAPTER_INFO结构体指针存储本机网卡信息
PIP_ADAPTER_INFO pIpAdapterInfo = new IP_ADAPTER_INFO();
PIP_ADAPTER_INFO pAdapter = NULL;
//得到结构体大小,用于GetAdaptersInfo参数
unsigned long stSize = sizeof(IP_ADAPTER_INFO);
//调用GetAdaptersInfo函数,填充pIpAdapterInfo指针变量;其中stSize参数既是一个输入量也是一个输出量
int nRel = GetAdaptersInfo(pIpAdapterInfo,&stSize);
//记录网卡数量
int netCardNum = 0;
//记录每张网卡上的IP地址数量
int IPnumPerNetCard = 0;
if (ERROR_BUFFER_OVERFLOW == nRel)
{
//如果函数返回的是ERROR_BUFFER_OVERFLOW
//则说明GetAdaptersInfo参数传递的内存空间不够,同时其传出stSize,表示需要的空间大小
//这也是说明为什么stSize既是一个输入量也是一个输出量
//释放原来的内存空间
delete pIpAdapterInfo;
//重新申请内存空间用来存储所有网卡信息
pIpAdapterInfo = (PIP_ADAPTER_INFO)new BYTE[stSize];
//再次调用GetAdaptersInfo函数,填充pIpAdapterInfo指针变量
nRel=GetAdaptersInfo(pIpAdapterInfo,&stSize);    
}
if (ERROR_SUCCESS == nRel)
{
//输出网卡信息
//可能有多网卡,因此通过循环去判断
pAdapter = pIpAdapterInfo;
while (pAdapter)
{
cout<<"网卡数量:"<<++netCardNum<<endl;
cout<<"网卡名称:"<<pAdapter->AdapterName<<endl;
cout<<"网卡描述:"<<pAdapter->Description<<endl;
switch(pAdapter->Type)
{
case MIB_IF_TYPE_OTHER:
cout<<"网卡类型:"<<"OTHER"<<endl;
break;
case MIB_IF_TYPE_ETHERNET:
cout<<"网卡类型:"<<"ETHERNET"<<endl;
break;
case MIB_IF_TYPE_TOKENRING:
cout<<"网卡类型:"<<"TOKENRING"<<endl;
break;
case MIB_IF_TYPE_FDDI:
cout<<"网卡类型:"<<"FDDI"<<endl;
break;
case MIB_IF_TYPE_PPP:
printf("PP\n");
cout<<"网卡类型:"<<"PPP"<<endl;
break;
case MIB_IF_TYPE_LOOPBACK:
cout<<"网卡类型:"<<"LOOPBACK"<<endl;
break;
case MIB_IF_TYPE_SLIP:
cout<<"网卡类型:"<<"SLIP"<<endl;
break;
default:
break;
}
cout<<"网卡MAC地址:";
for (DWORD i = 0; i < pAdapter->AddressLength; i++)
if (i < pAdapter->AddressLength-1)
{
printf("%02X-", pAdapter->Address[i]);
}
else
{
printf("%02X\n", pAdapter->Address[i]);
}
cout<<"网卡IP地址如下:"<<endl;
//可能网卡有多IP,因此通过循环去判断
IP_ADDR_STRING *pIpAddrString =&(pAdapter->IpAddressList);
IPnumPerNetCard = 0;
do 
{
cout<<"该网卡上的IP数量:"<<++IPnumPerNetCard<<endl;
cout<<"IP 地址:"<<pIpAddrString->IpAddress.String<<endl;
cout<<"子网地址:"<<pIpAddrString->IpMask.String<<endl;
cout<<"网关地址:"<<pAdapter->GatewayList.IpAddress.String<<endl;
pIpAddrString=pIpAddrString->Next;
} while (pIpAddrString);
if(getAdapterState(pAdapter->Index))
cout<<"网卡工作正常"<<endl;
else
cout<<"网卡工作异常"<<endl;
pAdapter = pAdapter->Next;
cout<<"--------------------------------------------------------------------"<<endl;
}
}
//释放内存空间
if (pIpAdapterInfo)
{
delete []pIpAdapterInfo;
pIpAdapterInfo=NULL;
}
return 0;
}

运行界面如下:


关于本文更为详细的说明及代码示例详见:http://download.csdn.net/detail/lusirking/9562517

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章