C/C++ 获取硬件唯一标识 (CPU序列号/硬盘号)

调用控制台来获取CPU序列号/硬盘号
不同PC的硬盘数量可能不同 若有多个硬盘,此处的输出将多个硬盘号拼接到了一起 若需拆分可自行修改。
注:插拔硬盘U盘等会造成获取到的硬盘号不同,如果想以此为机器码的同学可用BIOS序列号替代硬盘号。命令行调用改为wmic bios get serialnumber 即可

#if !defined(AFX_14BEC153_17B9_47BE_845F_71A27BF26B59__INCLUDED_)  
#define AFX_14BEC153_17B9_47BE_845F_71A27BF26B59__INCLUDED_  

#if _MSC_VER > 1000  
#pragma once  
#endif // _MSC_VER > 1000  

#include <iostream>  
#include <string>  
#include <windows.h>  
//#define CPUSIZE 17

using namespace std;

//--------------------------------------------------------------  
//                      CPU序列号  
//--------------------------------------------------------------  
BOOL GetCpuByCmd(string &ider, int len = 128);

BOOL GetDiskByCmd(string &ider, int len = 128);


#endif // !defined(AFX_14BEC153_17B9_47BE_845F_71A27BF26B59__INCLUDED_)  


//--------------------------------------------------------------  
//                      CPU序列号  
//--------------------------------------------------------------  
BOOL GetCpuByCmd(string &ider, int len/*=128*/)
{
	//CPU序列
	const long MAX_COMMAND_SIZE = 10000; // 命令行输出缓冲大小     
	WCHAR szFetCmd[] = L"wmic cpu get processorid"; // 获取CPU序列号命令行    
	const string strEnSearch = "ProcessorId"; // CPU序列号的前导信息  


	BOOL   bret = FALSE;
	HANDLE hReadPipe = NULL; //读取管道  
	HANDLE hWritePipe = NULL; //写入管道      
	PROCESS_INFORMATION pi;   //进程信息      
	STARTUPINFO         si;   //控制命令行窗口信息  
	SECURITY_ATTRIBUTES sa;   //安全属性  

	char            szBuffer[MAX_COMMAND_SIZE + 1] = { 0 }; // 放置命令行结果的输出缓冲区  
	string          strBuffer;
	unsigned long   count = 0;
	long            ipos = 0;

	memset(&pi, 0, sizeof(pi));
	memset(&si, 0, sizeof(si));
	memset(&sa, 0, sizeof(sa));

	pi.hProcess = NULL;
	pi.hThread = NULL;
	si.cb = sizeof(STARTUPINFO);
	sa.nLength = sizeof(SECURITY_ATTRIBUTES);
	sa.lpSecurityDescriptor = NULL;
	sa.bInheritHandle = TRUE;

	//1.0 创建管道  
	bret = CreatePipe(&hReadPipe, &hWritePipe, &sa, 0);
	if (!bret)
	{
		goto END;
	}

	//2.0 设置命令行窗口的信息为指定的读写管道  
	GetStartupInfo(&si);
	si.hStdError = hWritePipe;
	si.hStdOutput = hWritePipe;
	si.wShowWindow = SW_HIDE; //隐藏命令行窗口  
	si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;

	//3.0 创建获取命令行的进程  
	bret = CreateProcess(NULL, szFetCmd, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi);
	if (!bret)
	{
		goto END;
	}

	

	//4.0 读取返回的数据  
	WaitForSingleObject(pi.hProcess, 500/*INFINITE*/);
	bret = ReadFile(hReadPipe, szBuffer, MAX_COMMAND_SIZE, &count, 0);
	if (!bret)
	{
		goto END;
	}

	//5.0 查找CPU序列号  
	bret = FALSE;
	strBuffer = szBuffer;
	ipos = strBuffer.find(strEnSearch);

	if (ipos < 0) // 没有找到  
	{
		goto END;
	}
	else
	{
		strBuffer = strBuffer.substr(ipos + strEnSearch.length());
	}

	memset(szBuffer, 0x00, sizeof(szBuffer));
	strcpy_s(szBuffer, strBuffer.c_str());

	//modify here
	//去掉中间的空格 \r \n     
	char temp[512];
	memset(temp, 0, sizeof(temp));

	int index = 0;
	for (size_t i = 0; i < strBuffer.size(); i++)
	{
		if (strBuffer[i] != ' '&&strBuffer[i] != '\n'&&strBuffer[i] != '\r')
		{
			temp[index] = strBuffer[i];
			index++;
		}
	}
	ider = temp;

	bret = TRUE;

END:
	//关闭所有的句柄  
	CloseHandle(hWritePipe);
	CloseHandle(hReadPipe);
	CloseHandle(pi.hProcess);
	CloseHandle(pi.hThread);

	return(bret);
}

//--------------------------------------------------------------  
//                      硬盘编号 
//--------------------------------------------------------------  
BOOL GetDiskByCmd(string &ider, int len/*=128*/)
{
	//diskdrive
	const long MAX_COMMAND_SIZE = 10000; // 命令行输出缓冲大小     
	WCHAR szFetCmd[] = L"wmic diskdrive get serialnumber"; // 获取DiskDrive命令行    
	const string strEnSearch = "SerialNumber"; // DiskDrive序列号的前导信息  


	BOOL   bret = FALSE;
	HANDLE hReadPipe = NULL; //读取管道  
	HANDLE hWritePipe = NULL; //写入管道      
	PROCESS_INFORMATION pi;   //进程信息      
	STARTUPINFO         si;   //控制命令行窗口信息  
	SECURITY_ATTRIBUTES sa;   //安全属性  

	char            szBuffer[MAX_COMMAND_SIZE + 1] = { 0 }; // 放置命令行结果的输出缓冲区  
	string          strBuffer;
	unsigned long   count = 0;
	long            ipos = 0;

	memset(&pi, 0, sizeof(pi));
	memset(&si, 0, sizeof(si));
	memset(&sa, 0, sizeof(sa));

	pi.hProcess = NULL;
	pi.hThread = NULL;
	si.cb = sizeof(STARTUPINFO);
	sa.nLength = sizeof(SECURITY_ATTRIBUTES);
	sa.lpSecurityDescriptor = NULL;
	sa.bInheritHandle = TRUE;

	//1.0 创建管道  
	bret = CreatePipe(&hReadPipe, &hWritePipe, &sa, 0);
	if (!bret)
	{
		goto END;
	}

	//2.0 设置命令行窗口的信息为指定的读写管道  
	GetStartupInfo(&si);
	si.hStdError = hWritePipe;
	si.hStdOutput = hWritePipe;
	si.wShowWindow = SW_HIDE; //隐藏命令行窗口  
	si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;

	//3.0 创建获取命令行的进程  
	bret = CreateProcess(NULL, szFetCmd, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi);
	if (!bret)
	{
		goto END;
	}



	//4.0 读取返回的数据  
	WaitForSingleObject(pi.hProcess, 500/*INFINITE*/);
	bret = ReadFile(hReadPipe, szBuffer, MAX_COMMAND_SIZE, &count, 0);
	if (!bret)
	{
		goto END;
	}

	//5.0 查找CPU序列号  
	bret = FALSE;
	strBuffer = szBuffer;
	ipos = strBuffer.find(strEnSearch);

	if (ipos < 0) // 没有找到  
	{
		goto END;
	}
	else
	{
		strBuffer = strBuffer.substr(ipos + strEnSearch.length());
	}

	memset(szBuffer, 0x00, sizeof(szBuffer));
	strcpy_s(szBuffer, strBuffer.c_str());

	//modify here
	//去掉中间的空格 \r \n     
	char temp[512];
	memset(temp, 0, sizeof(temp));

	int index = 0;
	for (size_t i = 0; i < strBuffer.size(); i++)
	{
		if (strBuffer[i] != ' '&&strBuffer[i] != '\n'&&strBuffer[i] != '\r')
		{
			temp[index] = strBuffer[i];
			index++;
		}
	}
	ider = temp;

	bret = TRUE;

END:
	//关闭所有的句柄  
	CloseHandle(hWritePipe);
	CloseHandle(hReadPipe);
	CloseHandle(pi.hProcess);
	CloseHandle(pi.hThread);

	return(bret);
}



int main()
{
	string cpuid;
	
	if (!GetCpuByCmd(cpuid))
	{
		cout << "get cpu id failed!\n";
		return false;
	}
	cout << "cpu ID: " << cpuid << endl;
	
	string diskid;
	if (!GetDiskByCmd(diskid))
	{
		cout << "get diskdrive failed!\n";
		return false;
	}
	cout << "disk ID: " << diskid << endl;



	
	system("pause");
	return 0;
}
在C/C++获取计算机的硬盘序列号和显卡序列号是一个稍微复杂的任务,因为C/C++本身并没有提供直接的函数或API来获取这些信息。以下是一种可能的实现方式: 获取硬盘序列号: 在Windows系统上,你可以使用WMI接口来获取硬盘序列号。具体来说,可以使用`Win32_DiskDrive`类和`SerialNumber`属性来获取硬盘序列号。你可以使用C/C++调用WMI接口的方式来实现。 在Linux系统上,你可以通过读取设备文件`/dev/sd*`来获取硬盘序列号。具体来说,你可以打开设备文件,然后使用`HDIO_GET_IDENTITY` ioctl命令来获取硬盘的身份信息,其中包括序列号获取显卡序列号: 在Windows系统上,你可以使用WMI接口来获取显卡序列号。具体来说,可以使用`Win32_VideoController`类和`PNPDeviceID`属性来获取显卡的唯一标识符。同样,你可以通过C/C++调用WMI接口的方式来实现。 在Linux系统上,你可以通过读取显卡设备文件或者执行特定命令来获取显卡信息。具体的实现方式可能因为显卡类型和操作系统的差异而有所不同。你可以尝试使用命令行工具如`lspci`或者读取设备文件`/sys/class/drm/card*/device/uevent`来获取显卡信息。 请注意,以上提到的方法可能需要特权或系统级的操作,并且在不同的操作系统上实现方式可能有所差异。在实际应用中,建议先了解目标平台的相关文档和API,以确定最适合你的环境的方法来获取硬盘序列号和显卡序列号
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值