有需求才有创造,有了问题才会想着去解决,那么我这里的获取MAC地址的第4种方法也是在这种情况下产生的。因为公司有一个服务器产品,要写一个注册模块,而注册模块需要获取硬件信息,而硬件信息有很多,可以是硬盘序列号,CPU序列号,和网卡MAC,我首先使用的是硬盘序列号,因为GOOGLE一下一大堆,我要感谢所有这些无私奉献自己智慧结晶的可爱的人。
很快地写完一个注册模块,经过几个PC机,笔记本的测试,一切OK,以为可以run anywhere了,就提交给项目组了,以为万事大吉,过了很长时间,产品出炉了,拿出去用了,可以用户那边根本取不到机器信息。刚开始还纳闷,但是后来得知用户那边用的是服务器是SCIS硬盘。后来还是找不到既能获取普通硬盘的序列号也能获取SCIS硬盘的序列号通用的资料。
看了这个文章,我又很快地写好了一个注册模块,我用的是第1种方法,是由网卡的MAC为基本信息的,这次我先还是在几个PC机上做测试,不过这次比较幸运,问题马上就出现了,在一个装有防火墙软件的机器上无法获取MAC,原因cker已经说明了,是防火墙将文件共享服务关闭了,这样就获取不了MAC地址。
而使用第二种,根本就不是那回事,我在自己的开发机上就出事了,当然也不能用了。
第三种方法用的事SNMP,但不是每个机器都是安装了这个协议的,因此也不是一种通用的好方法。
在三种方法都没有办法行得通的情况下,一天我在看一本LINUX书的时候看到了管道,可以将输出重定向到管道,哈哈,这个时候也就想到了,用命令行config /all来获取网卡相关的信息,然后我重定向到管道,就可以获取各种和网卡相关的信息了,哈哈,于是这第四种方法也就出炉了。
这种方法的有点是你只要能够在机器上执行“ipconfig /all”命令你就可以得到MAC地址,而这个命令在所有的网卡可用的情况下都是可用的。
另外你要遍历所有的网卡MAC,那么只需提取出所有的MAC段即可。
程序的过程也清晰简单:
1. 创建一个无名管道。
2. 创建一个IPCONFIG 的进程,并将输出重定向到管道。
3. 从管道获取命令行返回的所有信息放入缓冲区lpszBuffer。
4. 从缓冲区lpszBuffer中获得抽取出MAC地址串。
通过我这个列子你也可以用来获取可以通过命令行方式来取得的信息,比如说主机名,IP地址等等。
源代码:
文件 GetMac.cpp:
// // 描述: 通过命令行方式得到MAC地址 // 作者: 郭洪锋 // 日期: 2005年7月1日
// email: guohongfeng@gmail.com / #include <string> #include <iostream> using namespace std;
//命令行输出缓冲大小 const long MAX_COMMAND_SIZE = 10000;
//获取MAC命令行 char szFetCmd[] = "ipconfig /all"; //网卡MAC地址的前导信息 const string str4Search = "Physical Address. . . . . . . . . : ";
//用命令行方式获取网卡MAC地址 BOOL GetMacByCmd(char *lpszMac);
// 函数名: GetMacByCmd(char *lpszMac) // 参数: // 输入: void // 输出: lpszMac,返回的MAC地址串 // 返回值: // TRUE: 获得MAC地址。 // FALSE: 获取MAC地址失败。 // 过程: // 1. 创建一个无名管道。 // 2. 创建一个IPCONFIG 的进程,并将输出重定向到管道。 // 3. 从管道获取命令行返回的所有信息放入缓冲区lpszBuffer。 // 4. 从缓冲区lpszBuffer中获得抽取出MAC串。 // // 提示:可以方便的由此程序获得IP地址等其他信息。 // 对于其他的可以通过其他命令方式得到的信息只需改变strFetCmd 和 // str4Search的内容即可。 ///
BOOL GetMacByCmd(char *lpszMac) { //初始化返回MAC地址缓冲区 memset(lpszMac, 0x00, sizeof(lpszMac)); BOOL bret;
SECURITY_ATTRIBUTES sa; HANDLE hReadPipe,hWritePipe;
sa.nLength = sizeof(SECURITY_ATTRIBUTES); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = TRUE; //创建管道 bret = CreatePipe(&hReadPipe, &hWritePipe, &sa, 0); if(!bret) { return FALSE; }
//控制命令行窗口信息 STARTUPINFO si; //返回进程信息 PROCESS_INFORMATION pi;
si.cb = sizeof(STARTUPINFO); GetStartupInfo(&si); si.hStdError = hWritePipe; si.hStdOutput = hWritePipe; si.wShowWindow = SW_HIDE; //隐藏命令行窗口 si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;
//创建获取命令行进程 bret = CreateProcess (NULL, szFetCmd, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi ); char szBuffer[MAX_COMMAND_SIZE+1]; //放置命令行输出缓冲区 string strBuffer;
if (bret) { WaitForSingleObject (pi.hProcess, INFINITE); unsigned long count; CloseHandle(hWritePipe);
memset(szBuffer, 0x00, sizeof(szBuffer)); bret = ReadFile(hReadPipe, szBuffer, MAX_COMMAND_SIZE, &count, 0); if(!bret) { //关闭所有的句柄 CloseHandle(hWritePipe); CloseHandle(pi.hProcess); CloseHandle(pi.hThread); CloseHandle(hReadPipe); return FALSE; } else { strBuffer = szBuffer; long ipos; ipos = strBuffer.find(str4Search);
//提取MAC地址串 strBuffer = strBuffer.substr(ipos+str4Search.length()); ipos = strBuffer.find("/n"); strBuffer = strBuffer.substr(0, ipos); } }
memset(szBuffer, 0x00, sizeof(szBuffer)); strcpy(szBuffer, strBuffer.c_str());
//去掉中间的“00-50-EB-0F-27-82”中间的'-'得到0050EB0F2782 int j = 0; for(int i=0; i<strlen(szBuffer); i++) { if(szBuffer[i] != '-') { lpszMac[j] = szBuffer[i]; j++; } }
//关闭所有的句柄 CloseHandle(hWritePipe); CloseHandle(pi.hProcess); CloseHandle(pi.hThread); CloseHandle(hReadPipe); return TRUE;
}
文件GetMacByCmd.cpp:
#include "stdafx.h" #include <iostream.h> #include <string>
#include "windows.h" using namespace std;
extern BOOL GetMacByCmd(char *lpszMac);
void main (int argc, char *argv[]) { char lpszMac[128]; memset(lpszMac, 0x00, sizeof(lpszMac));
//获取MAC GetMacByCmd(lpszMac);
//打印出MAC cout << lpszMac << endl;
//写入文件 FILE *fp = NULL; fp = fopen("c://1.txt", "w"); fwrite(lpszMac, sizeof(char), strlen(lpszMac), fp); fclose(fp); }