相关文章推荐
开朗的路灯  ·  idea的Tomcat的jmx ...·  1 年前    · 
满身肌肉的凉面  ·  VBA ...·  1 年前    · 
豪爽的啄木鸟  ·  转载:CMake I ...·  1 年前    · 

项目需要获取cpu的使用率,还需要跟任务管理器保持一致.

先是百度了一大圈,说是用GetSystemTimes这个api,来获取cpu总时间和空闲时间,然后做减法,每一段时间循环计算来实现的。

下面是代码:

__int64 CompareTime(FILETIME time1, FILETIME time2)
	__int64 a = time1.dwHighDateTime << 32 | time1.dwLowDateTime;
	__int64 b = time2.dwHighDateTime << 32 | time2.dwLowDateTime;
	return (b - a);
void TestDiskManager::GetCpuInfo() 
	FILETIME idleTime;//空闲时间 
	FILETIME kernelTime;//核心态时间 
	FILETIME userTime;//用户态时间 
	bool res;
	res = GetSystemTimes(&idleTime, &kernelTime, &userTime);
	cout << "GetSystemTimes(&, &, &) = " << res << endl << endl;
	HANDLE hEvent;
	FILETIME pre_idleTime;
	FILETIME pre_kernelTime;
	FILETIME pre_userTime;
	pre_idleTime = idleTime;
	pre_kernelTime = kernelTime;
	pre_userTime = userTime;
	hEvent = CreateEvent(NULL, FALSE, FALSE, NULL);
	//初始值为nonsignaled,并且每次触发后自动设置为nonsignaled
	while (1) {
		WaitForSingleObject(hEvent, 1000);//等待500毫秒
		res = GetSystemTimes(&idleTime, &kernelTime, &userTime);
		__int64 idle = CompareTime(pre_idleTime, idleTime);
		__int64 kernel = CompareTime(pre_kernelTime, kernelTime);
		__int64 user = CompareTime(pre_userTime, userTime);
		int cpu_occupancy_rate = (kernel + user - idle) * 100 / (kernel + user);
		//(总的时间 - 空闲时间)/ 总的时间 = 占用CPU时间的比率,即占用率
		int cpu_idle_rate = idle * 100 / (kernel + user);
		//空闲时间 / 总的时间 = 闲置CPU时间的比率,即闲置率 
		int cpu_kernel_rate = kernel * 100 / (kernel + user);
		//核心态时间 / 总的时间 = 核心态占用的比率 
		int cpu_user_rate = user * 100 / (kernel + user);
		//用户态时间 / 总的时间 = 用户态占用的比率 
		cout << left << setw(15) << "CPU占用率:" << cpu_occupancy_rate << "%" << endl
			<< setw(15) << "CPU闲置率:" << cpu_idle_rate << "%" << endl
			<< setw(15) << "核心态占比率:" << cpu_kernel_rate << "%" << endl
			<< setw(15) << "用户态占比率:" << cpu_user_rate << "%" << endl << endl;
		pre_idleTime = idleTime;
		pre_kernelTime = kernelTime;
		pre_userTime = userTime;
void TestDiskManager::GetCpuInfo2() 
    static FILETIME pre_idle_time;
    static FILETIME pre_kernel_time;
    static FILETIME pre_user_time;
    // 空闲时间
    FILETIME idle_time;
    // 内核时间
    FILETIME kernel_time;
    // 用户时间
    FILETIME user_time;
    BOOL ret = GetSystemTimes(&idle_time, &kernel_time, &user_time);
    int idle = CompareFileTime(&pre_idle_time, &idle_time);
    int kernel = CompareFileTime(&pre_kernel_time, &kernel_time);
    int user = CompareFileTime(&pre_user_time, &user_time);
    float rate = (kernel + user - idle) / (1.0 * (kernel + user));
    pre_idle_time = idle_time;
    pre_kernel_time = kernel_time;
    pre_user_time = user_time;
	cout << left << setw(15) << "CPU占用率:" << rate << "%" << endl;

  然后发现,跟任务管理器显示偏差很大,然后就去搜索原因,百度了一圈也没找到,都是类似的解决,没人说出个所以然来,最后在stackoverflow找到一个靠谱的答案,原文:

Why is this method to obtain CPU utilization produces results 2 times lower than that from the Task Manager?

相关截图:编辑

 大意是讲taskmgr(任务管理器)是超线程感知的。假设您有 2 个物理内核,总共有 4 个逻辑内核。将2个内核置于满负荷状态,任务列表将显示100%,而您的方法将显示50%,当所有4个逻辑内核都填满时,任务列表仍将显示100%,然后您的方法也将显示100%。

文中也没给解决方案,于是继续搜索。最后在stack大佬的帮助下,找到了这篇文章:

PerformanceCounter reporting higher CPU usage than what's observed

 大意是讲,使用了错误的参数。使用“Processor Information”而不是“Processor”。

结合之前找到的命令:

typeperf "\Processor(_Total)\% Processor Time"

稍作修改为:

typeperf "\Processor Information(_Total)\% Processor Utility"

试了下果然可行,数据什么的都很准确,如下图:

至此,基本上算是可以交差了。

不过在实现的过程中发现,做C++,面向百度编程已经不行了,必须要面向谷歌编程了。国内C++相关的干货实在太少,虽然天下文章一大抄,但是无脑抄就不太好了。

同名博客:一种通过代码(cmd命令行)准确获取cpu的使用率的方法_还在水里游的博客-CSDN博客