获取本地PC的IP地址和物理Mac地址(C++代码实现)
2017-11-30 17:11
1051 查看
本篇博客主要分为两部分,第一部分在c++环境下代码。代码功能为获取本地PC的IP地址和Mac地址。第二部分是程序的执行结果。
一、C++程序代码
#include <winsock2.h>//该头文件定义了Socket编程的功能
#include <stdio.h> //该头文件声明了输入输出流函数
#include <stdlib.h> //该头文件定义了一些通用函数
#include <httpext.h> //该头文件支持HTTP请求
#include <windef.h> //该头文件定义了Windows的所有数据基本型态
#include <Nb30.h> //该头文件声明了netbios的所有的函数
#pragma comment(lib,"ws2_32.lib")//连接ws2_32.lib库.只要程序中用到Winsock
API函数,都要用到 Ws2_32.lib
#pragma comment(lib,"netapi32.lib") //连接Netapi32.lib库,MAC获取中用到了NetApi32.DLL的功能
//在Win32平台上的Winsock编程都要经过下列的基本步骤:定义变量->获得Winsock版本->加载Winsock库->
//初始化->创建套接字->设置套接字选项->关闭套接字->卸载Winsock库,释放所有资源。
void CheckIP(void)//定义checkIP函数,用于取本机的ip地址
{
WSADATA wsaData;
char name[155];//定义用于存放获得主机名的变量
char *ip; //定义IP地址变量
PHOSTENT hostinfo;
//调用MAKEWORD()获得Winsocl版本的正确值,用于下面的加载Winscok库
//加载Winsock库,如果WSAStartup()函数返回值为0,说明加载成功,程序可以继续往下执行
if(WSAStartup(MAKEWORD(2, 0), &wsaData) == 0) //加载2.0socket库版本
{
//从主机名数据库中得到对应的主机名称,存入name变量中
//如果成功,将本地主机名存放入由name参数指定的缓冲区中
if(gethostname(name,
sizeof(name)) == 0)
{
//这是获取主机,如果获得主机名成功的话,将返回一个指针,指向hostinfo,hostinfo为PHOSTENT型的变量。
if((hostinfo = gethostbyname(name)) !=NULL)
{
//调用inet_ntoa()函数,
4000
将hostinfo结构变量中的h_addr_list转化为标准的IP地址(如202.197.11.12.)
ip =inet_ntoa(*(struct
in_addr *)*hostinfo->h_addr_list); //inet_addr()函数把地址串转换为IP地址
printf(" IP地址: %s/n", ip);//输出IP地址
}
}
WSACleanup(); //卸载Winsock库,并释放所有资源
}
}
//通过WindowsNT/Win2000中内置的NetApi32.DLL的功能来实现的。首先通过发送NCBENUM命令,获取网卡的
//数目和每张网卡的内部编号,然后对每个网卡标号发送NCBASTAT命令获取其MAC地址。
int getMAC(char *mac)//用NetAPI来获取网卡MAC地址
{
NCB ncb;
//定义一个NCB(网络控制块)类型的结构体变量ncb
typedef struct_ASTAT_ //自定义一个结构体_ASTAT_
{
ADAPTER_STATUS adapt;
NAME_BUFFER NameBuff[30];
}ASTAT, *PASTAT;
ASTAT Adapter;
typedef struct_LANA_ENUM //自定义一个结构体_ASTAT_
{
UCHAR length;
UCHAR lana[MAX_LANA];//存放网卡MAC地址
}LANA_ENUM;
LANA_ENUM lana_enum;
// 取得网卡信息列表
UCHAR uRetCode;
memset(&ncb,0, sizeof(ncb));
//将已开辟内存空间ncb
的值均设为值 0
memset(&lana_enum,0, sizeof(lana_enum));
//清空一个结构类型的变量lana_enum,赋值为0
//对结构体变量ncb赋值
ncb.ncb_command = NCBENUM;
//统计系统中网卡的数量
ncb.ncb_buffer = (unsigned
char*)&lana_enum; //ncb_buffer成员指向由LANA_ENUM结构填充的缓冲区
ncb.ncb_length = sizeof(LANA_ENUM); //向网卡发送NCBENUM命令,以获取当前机器的网卡信息,如有多少个网卡,每个网卡的编号(MAC地址)
uRetCode =Netbios(&ncb); //调用netbois(ncb)获取网卡序列号
if(uRetCode !=
NRC_GOODRET)
returnuRetCode;
//对每一个网卡,以其网卡编号为输入编号,获取其MAC地址
for (int lana =0; lana<lana_enum.length; lana++)
{
ncb.ncb_command= NCBRESET;
//对网卡发送NCBRESET命令,进行初始化
ncb.ncb_lana_num= lana_enum.lana[lana]; uRetCode = Netbios(&ncb);
}
if (uRetCode!=
NRC_GOODRET)
returnuRetCode;
// 准备取得接口卡的状态块取得MAC地址
memset(&ncb,0, sizeof(ncb));
ncb.ncb_command = NCBASTAT;
//对网卡发送NCBSTAT命令,获取网卡信息
ncb.ncb_lana_num =lana_enum.lana[0]; //指定网卡号,这里仅仅指定第一块网卡,通常为有线网卡
strcpy((char*)ncb.ncb_callname,"*");//远程系统名赋值为*
ncb.ncb_buffer = (unsigned
char*)&Adapter; //指定返回的信息存放的变量
ncb.ncb_length = sizeof(Adapter);
//接着发送NCBASTAT命令以获取网卡的信息
uRetCode =Netbios(&ncb); // 取得网卡的信息,并且如果网卡正常工作的话,返回标准的冒号分隔格式。
if(uRetCode !=
NRC_GOODRET)
return uRetCode;
//把网卡MAC地址格式转化为常用的16进制形式,输出到字符串mac中
sprintf(mac,
"%02X-%02X-%02X-%02X-%02X-%02X", Adapter.adapt.adapter_address[0],Adapter.adapt.adapter_address[1], Adapter.adapt.adapter_address[2],Adapter.adapt.adapter_address[3], Adapter.adapt.adapter_address[4],Adapter.adapt.adapter_address[5]);
return 0;
}
int main(void) //主函数,程序的入口
{
CheckIP(); //调用CheckIP()函数获得,输出IP地址
char mac[200];
getMAC(mac); //调用getMAC()函数获得,输出MAC地址
printf(" mac地址 : %s /n",mac);
//getchar();//暂停控制台窗口
system("PAUSE");//暂停控制台窗口
return 0;
}
二、程序运行结果
下图为调用函数CheckIP()获得本地IP,调用函数getMAC()获得本地Mac地址,输出的结果如下所示:
![](https://img-blog.csdn.net/20171130165932853?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbmFpYm96aHVhbjM3NDQ=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center)
参考内容:
http://blog.csdn.net/bodybo/article/details/5768405 http://blog.csdn.net/clemontine/article/details/53141041
一、C++程序代码
#include <winsock2.h>//该头文件定义了Socket编程的功能
#include <stdio.h> //该头文件声明了输入输出流函数
#include <stdlib.h> //该头文件定义了一些通用函数
#include <httpext.h> //该头文件支持HTTP请求
#include <windef.h> //该头文件定义了Windows的所有数据基本型态
#include <Nb30.h> //该头文件声明了netbios的所有的函数
#pragma comment(lib,"ws2_32.lib")//连接ws2_32.lib库.只要程序中用到Winsock
API函数,都要用到 Ws2_32.lib
#pragma comment(lib,"netapi32.lib") //连接Netapi32.lib库,MAC获取中用到了NetApi32.DLL的功能
//在Win32平台上的Winsock编程都要经过下列的基本步骤:定义变量->获得Winsock版本->加载Winsock库->
//初始化->创建套接字->设置套接字选项->关闭套接字->卸载Winsock库,释放所有资源。
void CheckIP(void)//定义checkIP函数,用于取本机的ip地址
{
WSADATA wsaData;
char name[155];//定义用于存放获得主机名的变量
char *ip; //定义IP地址变量
PHOSTENT hostinfo;
//调用MAKEWORD()获得Winsocl版本的正确值,用于下面的加载Winscok库
//加载Winsock库,如果WSAStartup()函数返回值为0,说明加载成功,程序可以继续往下执行
if(WSAStartup(MAKEWORD(2, 0), &wsaData) == 0) //加载2.0socket库版本
{
//从主机名数据库中得到对应的主机名称,存入name变量中
//如果成功,将本地主机名存放入由name参数指定的缓冲区中
if(gethostname(name,
sizeof(name)) == 0)
{
//这是获取主机,如果获得主机名成功的话,将返回一个指针,指向hostinfo,hostinfo为PHOSTENT型的变量。
if((hostinfo = gethostbyname(name)) !=NULL)
{
//调用inet_ntoa()函数,
4000
将hostinfo结构变量中的h_addr_list转化为标准的IP地址(如202.197.11.12.)
ip =inet_ntoa(*(struct
in_addr *)*hostinfo->h_addr_list); //inet_addr()函数把地址串转换为IP地址
printf(" IP地址: %s/n", ip);//输出IP地址
}
}
WSACleanup(); //卸载Winsock库,并释放所有资源
}
}
//通过WindowsNT/Win2000中内置的NetApi32.DLL的功能来实现的。首先通过发送NCBENUM命令,获取网卡的
//数目和每张网卡的内部编号,然后对每个网卡标号发送NCBASTAT命令获取其MAC地址。
int getMAC(char *mac)//用NetAPI来获取网卡MAC地址
{
NCB ncb;
//定义一个NCB(网络控制块)类型的结构体变量ncb
typedef struct_ASTAT_ //自定义一个结构体_ASTAT_
{
ADAPTER_STATUS adapt;
NAME_BUFFER NameBuff[30];
}ASTAT, *PASTAT;
ASTAT Adapter;
typedef struct_LANA_ENUM //自定义一个结构体_ASTAT_
{
UCHAR length;
UCHAR lana[MAX_LANA];//存放网卡MAC地址
}LANA_ENUM;
LANA_ENUM lana_enum;
// 取得网卡信息列表
UCHAR uRetCode;
memset(&ncb,0, sizeof(ncb));
//将已开辟内存空间ncb
的值均设为值 0
memset(&lana_enum,0, sizeof(lana_enum));
//清空一个结构类型的变量lana_enum,赋值为0
//对结构体变量ncb赋值
ncb.ncb_command = NCBENUM;
//统计系统中网卡的数量
ncb.ncb_buffer = (unsigned
char*)&lana_enum; //ncb_buffer成员指向由LANA_ENUM结构填充的缓冲区
ncb.ncb_length = sizeof(LANA_ENUM); //向网卡发送NCBENUM命令,以获取当前机器的网卡信息,如有多少个网卡,每个网卡的编号(MAC地址)
uRetCode =Netbios(&ncb); //调用netbois(ncb)获取网卡序列号
if(uRetCode !=
NRC_GOODRET)
returnuRetCode;
//对每一个网卡,以其网卡编号为输入编号,获取其MAC地址
for (int lana =0; lana<lana_enum.length; lana++)
{
ncb.ncb_command= NCBRESET;
//对网卡发送NCBRESET命令,进行初始化
ncb.ncb_lana_num= lana_enum.lana[lana]; uRetCode = Netbios(&ncb);
}
if (uRetCode!=
NRC_GOODRET)
returnuRetCode;
// 准备取得接口卡的状态块取得MAC地址
memset(&ncb,0, sizeof(ncb));
ncb.ncb_command = NCBASTAT;
//对网卡发送NCBSTAT命令,获取网卡信息
ncb.ncb_lana_num =lana_enum.lana[0]; //指定网卡号,这里仅仅指定第一块网卡,通常为有线网卡
strcpy((char*)ncb.ncb_callname,"*");//远程系统名赋值为*
ncb.ncb_buffer = (unsigned
char*)&Adapter; //指定返回的信息存放的变量
ncb.ncb_length = sizeof(Adapter);
//接着发送NCBASTAT命令以获取网卡的信息
uRetCode =Netbios(&ncb); // 取得网卡的信息,并且如果网卡正常工作的话,返回标准的冒号分隔格式。
if(uRetCode !=
NRC_GOODRET)
return uRetCode;
//把网卡MAC地址格式转化为常用的16进制形式,输出到字符串mac中
sprintf(mac,
"%02X-%02X-%02X-%02X-%02X-%02X", Adapter.adapt.adapter_address[0],Adapter.adapt.adapter_address[1], Adapter.adapt.adapter_address[2],Adapter.adapt.adapter_address[3], Adapter.adapt.adapter_address[4],Adapter.adapt.adapter_address[5]);
return 0;
}
int main(void) //主函数,程序的入口
{
CheckIP(); //调用CheckIP()函数获得,输出IP地址
char mac[200];
getMAC(mac); //调用getMAC()函数获得,输出MAC地址
printf(" mac地址 : %s /n",mac);
//getchar();//暂停控制台窗口
system("PAUSE");//暂停控制台窗口
return 0;
}
二、程序运行结果
下图为调用函数CheckIP()获得本地IP,调用函数getMAC()获得本地Mac地址,输出的结果如下所示:
参考内容:
http://blog.csdn.net/bodybo/article/details/5768405 http://blog.csdn.net/clemontine/article/details/53141041
相关文章推荐
- android 获取本机的IP地址和mac物理地址的实现方法
- php/js获取客户端mac地址的实现代码_PHP教程
- C++获取本地mac地址和网关mac地址方法
- 分享:PHP获取MAC地址的实现代码
- android获取本机的IP地址和mac物理地址
- php/js获取客户端mac地址的实现代码
- 获取硬件信息的delphi源码(CPUID、操作系统、Mac物理地址、计算机名称、IP地址、用户名)
- C++之获取网卡物理地址(MAC)
- JS获取客户端网卡物理地址(MAC) 代码
- 用纯Java代码根据IP获取windows和linux的MAC物理地址
- C#获取路由器外网IP,MAC地址的实现代码
- C#获取本地计算机名,IP,MAC(物理)地址
- C++编码获取本地主机的MAC地址
- linux系统下C语言获取和设置本地MAC地址的代码
- JDK1.6 获取操作系统IP地址及对应物理mac地址
- php/js获取客户端mac地址的实现代码
- Linux 接口地址的获取-暨获取本地IP地址(所有地址,包括IPv4,IPV6,MAC 地址)
- C# 获取本地IP客户端IP地址以及MAC地址
- php/js获取客户端mac地址的实现代码
- 获取Mac、CPUID、硬盘序列号、本地IP地址、外网IP地址OCX控件