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) 
		   { 
			//这是获取主机,如果获得主机名成功的话,将返回一个指针,指向hostinfo,hostinfo为PHOSTENT型的变量。
			ip = inet_ntoa (*(struct in_addr *)*hostinfo->h_addr_list); 
			//inet_addr()函数把地址串转换为IP地址
			//调用inet_ntoa()函数,将hostinfo结构变量中的h_addr_list转化为标准的IP地址(如202.197.11.12.)
			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_      
	{
		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;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值