方法1:通过GetAdaptersInfo
- #include<winsock2.h>
- #include<Iphlpapi.h>
- #include<stdio.h>
- #pragma comment(lib,"Iphlpapi.lib")
- int main()
- {
- PIP_ADAPTER_INFO pAdapterInfo;
- PIP_ADAPTER_INFO pAdapter = NULL;
- DWORD dwRetVal = 0;
- ULONG ulOutBufLen;
- pAdapterInfo=(PIP_ADAPTER_INFO)malloc(sizeof(IP_ADAPTER_INFO));
- ulOutBufLen = sizeof(IP_ADAPTER_INFO);
- // 第一次调用GetAdapterInfo获取ulOutBufLen大小
- if (GetAdaptersInfo( pAdapterInfo, &ulOutBufLen) == ERROR_BUFFER_OVERFLOW)
- {
- free(pAdapterInfo);
- pAdapterInfo = (IP_ADAPTER_INFO *) malloc (ulOutBufLen);
- }
- if ((dwRetVal = GetAdaptersInfo( pAdapterInfo, &ulOutBufLen)) == NO_ERROR) {
- pAdapter = pAdapterInfo;
- while (pAdapter)
- {
- printf("Adapter Name: \t%s\n", pAdapter->AdapterName);
- printf("Adapter Desc: \t%s\n", pAdapter->Description);
- printf("MAC Addr: \t%02x-%02x-%02x-%02x-%02x-%02x\n",
- pAdapter->Address[0],
- pAdapter->Address[1],
- pAdapter->Address[2],
- pAdapter->Address[3],
- pAdapter->Address[4],
- pAdapter->Address[5]);
- printf("IP Address: \t%s\n", pAdapter->IpAddressList.IpAddress.String);
- printf("IP Mask: \t%s\n", pAdapter->IpAddressList.IpMask.String);
- printf("Gateway: \t%s\n", pAdapter->GatewayList.IpAddress.String);
- pAdapterpAdapter = pAdapter->Next;
- }
- }
- else
- {
- printf("Call to GetAdaptersInfo failed.\n");
- }
- }
GetAdaptersInfo函数是ms提供给程序员用来枚举网卡信息API函数,它被封装在MS SDK中,如果你安装了MSSDK在include 和lib文件夹下,可以找到iphlpapi.h 和对应的iphlpapi.lib。很遗憾,在VC6下你不的不面对一系列头文件中变量的声明冲突问题,我暂时还没有找到解决方案。不过有却有个用GetAdaptersInfo的成熟方法,使用动态链接。
为了使用动态链接,必须保证系统中存在 iphlpapi.dll 这个动态链接库.
GetAdaptersInfo 函数 在MSSDK中声明为
DWORD GetAdaptersInfo(PIP_ADAPTER_INFO,PULONG);
注意在声明此函数类型时,请在类型前加入PASCAL调用约定 __stdcall
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//点到点协议适配器 PPP协议接入网络
// * MIB_IF_TYPE_LOOPBACK 24//回环(Loopback)适配器
// * MIB_IF_TYPE_SLIP 28//串行适配器(Serial Line Interface Protocol)
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;//是否启用WINS(Windows Internet Name Service)
IP_ADDR_STRING PrimaryWinsServer;//主WINS地址
IP_ADDR_STRING SecondaryWinsServer;//辅WINS地址
time_t LeaseObtained;//向DHCP*****IP地址的时间,DhcpEnabled=TRUE时有效
time_t LeaseExpires;//向DHCP*****IP地址到期时间,DhcpEnabled=TRUE时有效
} 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;
typedef struct { char String[4 * 4]; } IP_ADDRESS_STRING, *PIP_ADDRESS_STRING, IP_MASK_STRING, *PIP_MASK_STRING;
再搜索MSDN将此结构中的三个宏和两个结构补全。
注意声明顺序。
整个IP_ADAPTER_INFO 结构的大小为 640byte。
注意调用GetAdaptersInfo时,如果第一个参数分配的内存不够,第二个参数将返回需要内存的大小,通过两次调用,就可以得到系统网卡信息。整个调用过程如下:
最后不要忘了释放空间和动态链接库..
delete[] pInfo;
FreeLibrary(hInst);