获取本地PC的IP地址和物理Mac地址(C++代码实现)

本篇博客主要分为两部分,第一部分在c++环境下代码。代码功能为获取本地PC的IP地址和Mac地址。第二部分是程序的执行结果。


一、C++程序代码

#include //该头文件定义了Socket编程的功能

#include    //该头文件声明了输入输出流函数

#include      //该头文件定义了一些通用函数

#include  //该头文件支持HTTP请求

#include      //该头文件定义了Windows的所有数据基本型态

#include     //该头文件声明了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,hostinfoPHOSTENT型的变量。

         if((hostinfo = gethostbyname(name)) !=NULL)

         {

//调用inet_ntoa()函数,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

   {

      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地址,输出的结果如下所示:

获取本地PC的IP地址和物理Mac地址(C++代码实现)_第1张图片


参考内容:

http://blog.csdn.net/bodybo/article/details/5768405

http://blog.csdn.net/clemontine/article/details/53141041


你可能感兴趣的:(Socket编程)