(轉)通過WMI獲取網卡MAC地址、硬盤序列號、主板序列號、CPU ID、BIOS序列號


   最近由於項目的需要,需要在程序中獲取機器的硬盤序列號和MAC地址等信息,在C#下,可以很容易的獲得這些信息,但是在C++程序中感覺比較麻煩。經過百度,發現很多大蝦都是通過WMI來獲取這些硬件信息的,網上也有相關的代碼,通過實際調試,也發現確實可以通過WMI來獲取這些信息。前兩天,在網上突然搜到一位大牛寫的比較完整的程序,為了以后使用方便,就轉載記錄一下。同時,也會在大牛的代碼中增加一些自己的注釋,都是自己在實際使用過程中遇到的問題。

#define _WIN32_DCOM

#include "stdafx.h"
#include<iostream>
#include<fstream>
#include<string>
#include "direct.h"
#include <tchar.h>
#include <time.h>
#include <comdef.h>
#include <Wbemidl.h>
#include <conio.h>
#include "atlstr.h"
#include "atlbase.h"
//#include "TcpCtl.h"
//#include "winsock2.h"
//#include "InitDll.h"
using namespace std;

# pragma comment(lib, "wbemuuid.lib")
# pragma comment(lib, "ws2_32.lib")

//通過WMI獲取主板號
BOOL ManageWMIBord(char bord[])
{
HRESULT hres;
// Step 1:  初始化COM 
//hres = CoInitializeEx(0, COINIT_MULTITHREADED); //網上的代碼都是使用這行語句進行初始化,但是我在實際使用中,發現也可以采用下面的語句進行初始化
hres = CoInitialize(0);

//網上的代碼是沒有注釋下面這個判斷的,但是實際使用中發現,如果之前已經初始化成功了,在第二次初始化的時候,下面的代碼就會導致返回false,所以,實際使用中我就注釋掉了
//if (FAILED(hres))
//{
// cout << "Failed to initialize COM library. Error code = 0x"
// << hex << hres << endl;
// return false; // Program has failed.
//}


// Step 2:  設置COM的安全認證級別  

//在實際使用過程中,我發現如果這一步不注釋掉的話,程序總是返回false,注釋掉之后程序反而可以正常運行,原因未知

// Note: If you are using Windows 2000, you need to specify -
// the default authentication credentials for a user by using
// a SOLE_AUTHENTICATION_LIST structure in the pAuthList ----
// parameter of CoInitializeSecurity ------------------------
////hres = CoInitializeSecurity(
//// NULL,
//// -1, // COM authentication
//// NULL, // Authentication services
//// NULL, // Reserved
//// RPC_C_AUTHN_LEVEL_DEFAULT, // Default authentication
//// RPC_C_IMP_LEVEL_IMPERSONATE, // Default Impersonation
//// NULL, // Authentication info
//// EOAC_NONE, // Additional capabilities
//// NULL // Reserved
//// );
////
////if (FAILED(hres))
////{
//// cout << "Failed to initialize security. Error code = 0x"
//// << hex << hres << endl;
//// CoUninitialize();
//// return false; // Program has failed.
////}

// Step 3:  獲得WMI連接COM接口  
IWbemLocator *pLoc = NULL;
hres = CoCreateInstance(
CLSID_WbemLocator,
0,
CLSCTX_INPROC_SERVER,
IID_IWbemLocator, (LPVOID *) &pLoc);

if (FAILED(hres))
{
cout << "Failed to create IWbemLocator object."
<< " Err code = 0x"
<< hex << hres << endl;
CoUninitialize();
return false; // Program has failed.
}


// Step 4:  通過連接接口連接WMI的內核對象名"ROOT//CIMV2"  
IWbemServices *pSvc = NULL;

hres = pLoc->ConnectServer(

_bstr_t(L"ROOT\\CIMV2"), // Object path of WMI namespace
NULL, // User name. NULL = current user
NULL, // User password. NULL = current
0, // Locale. NULL indicates current
NULL, // Security flags.
0, // Authority (e.g. Kerberos)
0, // Context object
&pSvc // pointer to IWbemServices proxy
);

if (FAILED(hres))
{
cout << "Could not connect. Error code = 0x"
<< hex << hres << endl;
pLoc->Release();
CoUninitialize();
return false; // Program has failed.
}
//cout << "Connected to ROOT\\CIMV2 WMI namespace" << endl;

// Step 5:  設置請求代理的安全級別   
hres = CoSetProxyBlanket(
pSvc, // Indicates the proxy to set
RPC_C_AUTHN_WINNT, // RPC_C_AUTHN_xxx
RPC_C_AUTHZ_NONE, // RPC_C_AUTHZ_xxx
NULL, // Server principal name
RPC_C_AUTHN_LEVEL_CALL, // RPC_C_AUTHN_LEVEL_xxx
RPC_C_IMP_LEVEL_IMPERSONATE, // RPC_C_IMP_LEVEL_xxx
NULL, // client identity
EOAC_NONE // proxy capabilities
);
if (FAILED(hres))
{
cout << "Could not set proxy blanket. Error code = 0x"
<< hex << hres << endl;
pSvc->Release();
pLoc->Release();
CoUninitialize();
return false; // Program has failed.
}
// Step 6: 通過請求代理來向WMI發送請求----
// For example, get the name of the operating system
IEnumWbemClassObject* pEnumerator = NULL;
hres = pSvc->ExecQuery(
bstr_t("WQL"),
//bstr_t("SELECT * FROM Win32_NetworkAdapterConfiguration WHERE IPEnabled = 'TRUE'"),
bstr_t("SELECT * FROM Win32_BaseBoard"),//只需要通過修改這里的查詢語句,就可以實現對MAC地址等其他信息的查詢
WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY,
NULL,
&pEnumerator);

if (FAILED(hres))
{
cout << "Query for Network Adapter Configuration failed."
<< " Error code = 0x"
<< hex << hres << endl;
pSvc->Release();
pLoc->Release();
CoUninitialize();
return false; // Program has failed.
}


// Step 7:  循環枚舉所有的結果對象

IWbemClassObject *pclsObj;
ULONG uReturn = 0;

while (pEnumerator)
{
HRESULT hr = pEnumerator->Next(WBEM_INFINITE, 1,
&pclsObj, &uReturn);
if(0 == uReturn)
{
break;
}
VARIANT vtProp;
VariantInit(&vtProp);

hr = pclsObj->Get(L"SerialNumber", 0, &vtProp, 0, 0);//查詢不同的硬件信息,除了修改上面的查詢語句,這里的字段也要修改

if(!FAILED(hr))
{
CW2A tmpstr1(vtProp.bstrVal);
strcpy_s(bord,200,tmpstr1);//這里的200是可調的,自己根據實際情況設置,但是肯定不能大於bord的長度
//cout << "BordSN:" << sn << endl;

}

VariantClear(&vtProp);
}//end while

// 釋放資源  
pSvc->Release();
pLoc->Release();
pEnumerator->Release();
pclsObj->Release();
CoUninitialize();
return true; // Program successfully completed.

}

 

附:

// WQL查詢語句
const T_WQL_QUERY szWQLQuery[] = {
// 網卡原生MAC地址
"SELECT * FROM Win32_NetworkAdapter WHERE (MACAddress IS NOT NULL) AND (NOT (PNPDeviceID LIKE 'ROOT%'))",
L"PNPDeviceID",

// 硬盤序列號
"SELECT * FROM Win32_DiskDrive WHERE (SerialNumber IS NOT NULL) AND (MediaType LIKE 'Fixed hard disk%')",
L"SerialNumber",

// 主板序列號
"SELECT * FROM Win32_BaseBoard WHERE (SerialNumber IS NOT NULL)",
L"SerialNumber",

// 處理器ID
"SELECT * FROM Win32_Processor WHERE (ProcessorId IS NOT NULL)",
L"ProcessorId",

// BIOS序列號
"SELECT * FROM Win32_BIOS WHERE (SerialNumber IS NOT NULL)",
L"SerialNumber",

// 主板型號
"SELECT * FROM Win32_BaseBoard WHERE (Product IS NOT NULL)",
L"Product",

// 網卡當前MAC地址
"SELECT * FROM Win32_NetworkAdapter WHERE (MACAddress IS NOT NULL) AND (NOT (PNPDeviceID LIKE 'ROOT%'))",
L"MACAddress",

//當前機器的型號和廠商

"SELECT * FROM Win32_computersystem",

L"Manufacturer",L"Model"


}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM