用 Windows API “GetAdaptersInfo” 获取 MAC 时遇到的问题
前段时间有个项目需要获取客户端的 MAC 地址,用作统计去重的参考数据。从网上查到的获取 MAC 地址的代码,大多是用同一段代码修改的。于是我也用了那段代码。代码如下:
- void GetMAC(BYTE mac[BUF_SIZE])
- {
- ULONG size_pointer;
- PIP_ADAPTER_INFO pip_adapter_info = NULL;
- if(ERROR_BUFFER_OVERFLOW != GetAdaptersInfo(NULL, &size_pointer))
- {
- wsprintfA((LPSTR)mac, "GetMAC Failed! ErrorCode: %d", GetLastError());
- return;
- }
- pip_adapter_info = (PIP_ADAPTER_INFO)calloc(size_pointer, sizeof(BYTE));
- if(NULL == pip_adapter_info)
- {
- lstrcpyA((LPSTR)mac, "GetMAC Failed! Because calloc failed!");
- return;
- }
- if(ERROR_SUCCESS != GetAdaptersInfo(pip_adapter_info, &size_pointer))
- {
- wsprintfA((LPSTR)mac, "GetMAC Failed! ErrorCode: %d", GetLastError());
- free(pip_adapter_info);
- return;
- }
- for(int i = 0; i < 6; ++i)
- {
- wsprintfA((LPSTR)mac, "%02X", pip_adapter_info->Address[i]);
- mac += 2;
- }
- }
在自己的电脑上、虚拟机上测试了一番,没有发现问题,觉得一切 OK 之后将产品发布出去,结果发现许多机器返回的 MAC 为 NULL!!!
问题严重,又重新研究了一番,发现之所以 MAC 返回 NULL 是因为这段代码并不完整:用 GetAdaptersInfo 获取到的不一定是网卡信息,有可能是别的适配器信息,不是网卡就没有 MAC,结果当然为 NULL!花了点时间把代码完善了一下,加了个检测适配器特性的函数,完成了该功能,最终的代码如下:
- #include <windows.h>
- #include <iphlpapi.h> // API GetAdaptersInfo 头文件
- #include <shlwapi.h> // API StrCmpIA 头文件
- #pragma comment(lib, "iphlpapi.lib")
- #pragma comment(lib, "shlwapi.lib")
- #include <Strsafe.h> // API StringCbPrintfA 头文件
- #include <shellapi.h> // API lstrcpyA 头文件
- //
- // 功能:获取适配器特性
- // 参数:
- // adapter_name 适配器 ID
- // 返回值:成功则返回由参数指定的适配器的特性标志,是一个 DWORD 值,失败返回 0
- //
- UINT GetAdapterCharacteristics(char* adapter_name)
- {
- if(adapter_name == NULL || adapter_name[0] == 0)
- return 0;
- HKEY root = NULL;
- // 打开存储适配器信息的注册表根键
- if(ERROR_SUCCESS != RegOpenKeyExA(HKEY_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002BE10318}", 0, KEY_READ, &root))
- return 0;
- DWORD subkeys = 0;
- // 获取该键下的子键数
- if(ERROR_SUCCESS != RegQueryInfoKeyA(root, NULL, NULL, NULL, &subkeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL))
- subkeys = 100;
- DWORD ret_value = 0;
- for(DWORD i = 0; i < subkeys; i++)
- {
- // 每个适配器用一个子键存储,子键名为从 0 开始的 4 位数
- char subkey[MAX_SIZE];
- memset(subkey, 0, MAX_SIZE);
- StringCbPrintfA(subkey, MAX_SIZE, "%04u", i);
- // 打开该子键
- HKEY hKey = NULL;
- if(ERROR_SUCCESS != RegOpenKeyExA(root, subkey, 0, KEY_READ, &hKey))
- continue;
- // 获取该子键对应的适配器 ID,存于 name 中
- char name[MAX_PATH];
- DWORD type = 0;
- DWORD size = MAX_PATH;
- if(ERROR_SUCCESS != RegQueryValueExA(hKey, "NetCfgInstanceId", NULL, &type, (LPBYTE)name, &size))
- {
- RegCloseKey(hKey);
- continue;
- }
- // 对比该适配器 ID 是不是要获取特性的适配器 ID
- if(StrCmpIA(name, adapter_name) != 0)
- {
- RegCloseKey(hKey);
- continue;
- }
- // 读取该适配器的特性标志,该标志存储于值 Characteristics 中
- DWORD val = 0;
- size = 4;
- LSTATUS ls = RegQueryValueExA(hKey, "Characteristics", NULL, &type, (LPBYTE)&val, &size);
- RegCloseKey(hKey);
- if(ERROR_SUCCESS == ls)
- {
- ret_value = val;
- break;
- }
- }
- RegCloseKey(root);
- return ret_value;
- }
- //
- // 功能:获取 Mac 地址的二进制数据
- // 参数:
- // mac 用于输出 Mac 地址的二进制数据的缓冲区指针
- // 返回值:成功返回 mac 地址的长度,失败返回 0,失败时 mac 中保存一些简单的错误信息,可适当修改,用于调试
- //
- int GetMAC(BYTE mac[BUF_SIZE])
- {
- #define NCF_PHYSICAL 0x4
- DWORD AdapterInfoSize = 0;
- if(ERROR_BUFFER_OVERFLOW != GetAdaptersInfo(NULL, &AdapterInfoSize))
- {
- StringCbPrintfA((LPSTR)mac, BUF_SIZE, "GetMAC Failed! ErrorCode: %d", GetLastError());
- return 0;
- }
- void* buffer = malloc(AdapterInfoSize);
- if(buffer == NULL)
- {
- lstrcpyA((LPSTR)mac, "GetMAC Failed! Because malloc failed!");
- return 0;
- }
- PIP_ADAPTER_INFO pAdapt = (PIP_ADAPTER_INFO)buffer;
- if(ERROR_SUCCESS != GetAdaptersInfo(pAdapt, &AdapterInfoSize))
- {
- StringCbPrintfA((LPSTR)mac, BUF_SIZE, "GetMAC Failed! ErrorCode: %d", GetLastError());
- free(buffer);
- return 0;
- }
- int mac_length = 0;
- while(pAdapt)
- {
- if(pAdapt->AddressLength >= 6 && pAdapt->AddressLength <= 8)
- {
- memcpy(mac, pAdapt->Address, pAdapt->AddressLength);
- mac_length = pAdapt->AddressLength;
- UINT flag = GetAdapterCharacteristics(pAdapt->AdapterName);
- bool is_physical = ((flag & NCF_PHYSICAL) == NCF_PHYSICAL);
- if(is_physical)
- break;
- }
- pAdapt = pAdapt->Next;
- }
- free(buffer);
- return mac_length;
- }
- //
- // 功能:获取 Mac 地址,使用时直接调用此函数即可
- // 参数:
- // mac 用于存储 Mac 地址的缓冲区指针
- // 返回值:无返回值,函数执行完后会把 Mac 地址以16进制的形式存于参数指定的缓冲区中,若有错误,缓冲区中保存的是错误信息
- //
- void GetMacAddress( char* mac )
- {
- BYTE buf[BUF_SIZE];
- memset(buf, 0, BUF_SIZE);
- int len = GetMAC(buf);
- if(len <= 0)
- {
- lstrcpyA(mac, (LPCSTR)buf);
- return;
- }
- if(len == 6)
- StringCbPrintfA(mac, BUF_SIZE, "%02X-%02X-%02X-%02X-%02X-%02X", buf[0], buf[1], buf[2], buf[3], buf[4], buf[5]);
- else
- StringCbPrintfA(mac, BUF_SIZE, "%02X-%02X-%02X-%02X-%02X-%02X-%02X-%02X", buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6], buf[7]);
- }
编译环境: VS2008 + Windows SDK 7.1
函数功能在 Windows 2000、Windows XP、Windows 2003、Vista、 Win7 32位和 Win7 64 位下均测试通过。
用 Windows API “GetAdaptersInfo” 获取 MAC 时遇到的问题相关推荐
- php获取主板序列号,PHP获取通过windows系统命令wmic获取MAC地址、CPU序列号、主板序列号...
在项目中,客户需要系统在win系统上获取MAC地址.CPU序列号和主板序列号等,在网上搜索下,通过windows系统命令wmic可以获取,测试基本可行,HardwareInfo.php源代码如下: $ ...
- GetAdaptersInfo获取MAC地址
源代码: #include<atlbase.h> #include<atlconv.h> #include"iphlpapi.h" #pragma comm ...
- Windows API ——GetLogicalDriveStrings——获取逻辑驱动器
1 TCHAR buffer[MAX_PATH]; 2 DWORD dwVal=::GetLogicalDriveStrings(MAX_PATH,buffer); 3 4 CString str; ...
- C# 使用Windows API获取系统当前鼠标信息(图案)
通过使用Windows API来获取当前鼠标的图案,不论是系统图片还是自定义图标都能够获取到,在这个示例中,为了方便测试,给Form1添加了一个KeyPress事件,在程序激活状态下,将鼠标移动到任意 ...
- 获取mac地址方法之一 GetAdaptersInfo()
GetAdaptersInfo -20151116 防止返回的mac出现null 20151116 From:http://blog.csdn.net/weiyumingwww/article/det ...
- Windows/Linux获取Mac地址和CPU序列号实现
UUID(Universally Unique Identifier)即通用唯一标识符,是指在一台机器上生成的数字,保证在全球范围的唯一性.可用的开源库如libuuid,可参考https://blog ...
- Windows API的时间结构体、时间转换及时间获取
Windows API的时间结构体.时间转换及时间获取 时间格式:DosDateTime <====>FileTime <====> SystemTime | ...
- C# 获得窗体句柄并发送消息(利用windows API可在不同进程中获取)
C#使用Windows API获取窗口句柄控制其他程序窗口 编写程序模拟鼠标和键盘操作可以方便的实现你需要的功能,而不需要对方程序为你开放接口.比如,操作飞信定时发送短信等.我之前开发过飞信耗子,用的 ...
- C#通过Windows API捕获窗,获取窗口文本(FindWindow、GetWindowText),附录:Windows窗口消息大全、Windows API大全
文章目录 一.前言 二.使用Spy++工具分析窗口 三.C#通过Windows API捕获窗口,获取窗口文本 四.附录:Windows窗口消息 五.Windows API大全 1.API之网络函数 2 ...
最新文章
- 2022-2028年中国TAC薄膜行业市场全景评估及投资前景规划报告
- 03--MySQL自学教程:MySQL安装后的目录结构介绍和简单语法指令
- 集成学习算法之boosting、bagging和随机森林算法原理
- 网站搭建从零开始(四) 服务器的配置
- 图片优化_网站里的图片应该如何优化
- 人类首张黑洞照片咋拍的:7千TB数据太大网络传不了只能飞机运
- 华为云教你7天玩转电商应用性能调优,课程免费速来报名!
- FinTech领域的风险控制——策略篇
- 修改tomcat默认端口号
- html验证码 按住向右滑动,js实现滑动滑块验证登录
- tolua++ 参考手册
- 台式计算机配置单,最新台式电脑组装配置推荐
- RuoYi-flowable工作流管理
- 穿膜肽TAT修饰载荧光探针香豆素-6脂质体
- 北京折叠——一部刚要开始就已经结束的科幻小说
- 测试开发人员与开发人员_我是真正的开发人员还是优秀的Googler?
- 【联盛德W806上手笔记】六、7816/UART 控制器
- linux 查看dhcp dns,RHEL6 DNS+DHCP+DDNS
- mysql 页分裂_InnoDB中的页合并与分裂
- vue项目获取下拉框选中id_vue获取下拉框值