C++ socket实现获取本机IP&MAC

#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的功能
using namespace System;

void CheckIP(void)       //定义checkIP函数,用于取本机的ip地址
{
 WSADATA wsaData;
 char name[155];     
 char *ip;       
 PHOSTENT hostinfo;
 
 //调用MAKEWORD()获得Winsocl版本的正确值,用于下面的加载Winscok库
 if ( WSAStartup( MAKEWORD(2,0), &wsaData ) == 0 )
 {  
  //加载Winsock库,如果WSAStartup()函数返回值为0,说明加载成功,程序可以继续往下执行
  if( gethostname ( name, sizeof(name)) == 0)
  {
    //如果成功,将本地主机名存放入由name参数指定的缓冲区中
     if((hostinfo = gethostbyname(name)) != NULL)
     {
 i = 0;
 while(NiULL != hostinfo->h_addr_list[i])
 { 
     //这是获取主机,如果获得主机名成功的话,将返回一个指针,指向hostinfo,hostinfo为PHOSTENT型的变量。
            ip = inet_ntoa (*(struct in_addr *)*hostinfo->h_addr_list[i]);
        //inet_addr()函数把地址串转换为IP地址
        //调用inet_ntoa()函数,将hostinfo结构变量中的h_addr_list转化为标准的IP地址(如202.197.11.12.)
        printf(" IP地址: %s\n",ip);
     i++;
 }
     }
  }
  WSACleanup( );                         //卸载Winsock库,并释放所有资源
 }
}

//通过WindowsNT/Win2000中内置的NetApi32.DLL的功能来实现的。首先通过发送NCBENUM命令,获取网卡的
//数目和每张网卡的内部编号,然后对每个网卡标号发送NCBASTAT命令获取其MAC地址。
int getMAC(char * mac)                    //用NetAPI来获取网卡MAC地址
{    
 NCB ncb;              //定义一个NCB(网络控制块)类型的结构体变量ncb
 typedef struct _ASTAT_     
 {
  ADAPTER_STATUS   adapt;
  NAME_BUFFER   NameBuff   [30];    
 }ASTAT, *PASTAT;
 ASTAT Adapter;  

 typedef struct _LANA_ENUM    
 {
  UCHAR length;
  UCHAR lana[MAX_LANA];    
 }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)    
 return uRetCode;    

 //对每一个网卡,以其网卡编号为输入编号,获取其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)
  break;
 }
 if(uRetCode != NRC_GOODRET)
 return uRetCode;    
    
 // 准备取得接口卡的状态块取得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; 
 
 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()
{

    CheckIP();     
 char   mac[200];  
 getMAC(mac);
 printf(" mac地址 : %s \n",mac);

    return 0;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值