利用 GetSystemTimes 可以获得 Windows 系统的 Idle Time、 Kernel Time 和 User Time。Idle Time 是系统空闲的时间,也就是系统没有利用的时间。Kernel Time 是系统在内核模式下的执行时间。User Time 是系统在用户模式下的执行时间。由于 Kernel Time 包含了 Idle Time,因此 CPU 的总利用时间为 Kernel Time + User Time - Idle Time。间隔一段时间连续两次调用 GetSystemTimes,再利用时间差可以计算这段时间内的平均 CPU 总利用率。这个 API 返回的时间类型是 FILETIME,不能直接用其 dwHighDateTime 和 dwLowDateTime 成员相减。需要先将其转化为 uint64_t 再进行计算。下面是示例代码:

#include <Windows.h>
#include <iostream>
#include <thread>
#include <vector> void do_busy_things()
{
while (true) {
volatile int sum = 0;
for (volatile int i = 0; i < 10000; ++i) {
sum += i;
}
}
} void measure_cpu_usage()
{
struct system_time_t
{
uint64_t idle_time, kernel_time, user_time;
system_time_t operator-(system_time_t other) {
return {
.idle_time = idle_time - other.idle_time,
.kernel_time = kernel_time - other.kernel_time,
.user_time = user_time - other.user_time,
};
}
}; struct system_time_file_t
{
FILETIME idle_time, kernel_time, user_time;
system_time_t to_uint64_t() {
system_time_t time{};
time.idle_time = ((uint64_t)idle_time.dwHighDateTime << 32) | idle_time.dwLowDateTime;
time.kernel_time = ((uint64_t)kernel_time.dwHighDateTime << 32) | kernel_time.dwLowDateTime;
time.user_time = ((uint64_t)user_time.dwHighDateTime << 32) | user_time.dwLowDateTime;
return time;
}
}; while (true) {
system_time_file_t begin_time{}, end_time{};
GetSystemTimes(&begin_time.idle_time, &begin_time.kernel_time, &begin_time.user_time);
std::this_thread::sleep_for(std::chrono::seconds(1));
GetSystemTimes(&end_time.idle_time, &end_time.kernel_time, &end_time.user_time);
auto system_time = end_time.to_uint64_t() - begin_time.to_uint64_t();
double cpu_usage_percent = 1.0 * (system_time.kernel_time + system_time.user_time - system_time.idle_time) / (system_time.kernel_time + system_time.user_time) * 100;
std::cout << "cpu usage percent: " << cpu_usage_percent << "%" << std::endl;
}
} int main()
{
//std::vector<std::thread> v(std::thread::hardware_concurrency());
std::vector<std::thread> v(std::thread::hardware_concurrency() / 2);
std::thread t0(measure_cpu_usage); for (auto& t : v) {
t = std::thread(do_busy_things);
} for (auto& t : v) {
t.join();
} t0.join();
}

这段代码启动了物理线程数一半的线程执行 do_busy_things 函数,再启动一个线程测量 CPU 总利用率。可以观察到 CPU 总利用率在 50% ~ 52% 之间。打开任务管理器,显示 CPU 总利用率有 62%。使用 perfmon /res 打开资源监视器,在 CPU 一栏有 "62% CPU 使用率 118% 最大频率"。51% * 118% = 61%。任务管理器超频下的执行时间也包括在内,而 GetSystemTimes 得到的时间不包括超频下的执行时间。

将 do_busy_things 线程数改为 std::thread::hardware_concurrency(),也就是物理线程数。神奇的事情发生了,上面的代码显示 100%,任务管理器显示 100%,资源监视器显示 "116% CPU 使用率 116% 最大频率"。看来任务管理器进行了截断,如果 CPU 总利用率大于 100%,会直接显示100%。而资源监视器会实事求是地显示。

总的来说,使用 GetSystemTimes 计算 CPU 总利用率虽然没有包含超频执行时间,但是不会超过 100%,符合直觉。

Windows 平台计算 CPU 总利用率的更多相关文章

  1. 认识二进制安全与漏洞攻防技术 (Windows平台)

    二进制漏洞是指程序存在安全缺陷,导致攻击者恶意构造的数据(如Shellcode)进入程序相关处理代码时,改变程序原定的执行流程,从而实现破坏或获取超出原有的权限. 0Day漏洞 在计算机领域中,0da ...

  2. Windows平台分布式架构实践 - 负载均衡

    概述 最近.NET的世界开始闹腾了,微软官方终于加入到了对.NET跨平台的支持,并且在不久的将来,我们在VS里面写的代码可能就可以通过Mono直接在Linux和Mac上运行.那么大家(开发者和企业)为 ...

  3. [函数] Firemonkey Windows 重新计算 Font Baseline

    计算字型 Baseline 是一个不常用的函数,但如果想要显示不同大小文字下方对齐,就得用它来计算字型的 Baseline 才行,如果计算不准,显示的文字就会高高低低不整齐. 在 Firemonkey ...

  4. Windows平台分布式架构实践 - 负载均衡 上

    概述 最近.NET的世界开始闹腾了,微软官方终于加入到了对.NET跨平台的支持,并且在不久的将来,我们在VS里面写的代码可能就可以通过Mono直接在Linux和Mac上运行.那么大家(开发者和企业)为 ...

  5. Windows平台分布式网站系统应用(转)

    概述 最近.NET的世界开始闹腾了,微软官方终于加入到了对.NET跨平台的支持,并且在不久的将来,我们在VS里面写的代码可能就可以通过Mono直接在Linux和Mac上运行.那么大家(开发者和企业)为 ...

  6. Windows平台分布式架构-负载均衡(高并发)

    缘由 单纯想在winodows平台部署分布式程序,微软在IIS扩展的介绍中有涉及到Application Request Router + Web Farm + Url Rewriter可以实现分布式 ...

  7. Windows平台网站图片服务器架构的演进[转]

    构建在Windows平台之上的网站,往往会被业内众多架构师认为很“保守”.很大部分原因,是由于微软技术体系的封闭和部分技术人员的短视造成 的.由于长期缺乏开源支持,所以只能“闭门造车”,这样很容易形成 ...

  8. Windows平台分布式架构实践负载均衡

    Windows平台分布式架构实践 - 负载均衡 概述 最近.NET的世界开始闹腾了,微软官方终于加入到了对.NET跨平台的支持,并且在不久的将来,我们在VS里面写的代码可能就可以通过Mono直接在Li ...

  9. Windbg是windows平台上强大的调试器

    基础调试命令 - .dump/.dumpcap/.writemem/!runaway Windbg是windows平台上强大的调试器,它相对于其他常见的IDE集成的调试器有几个重要的优势, Windb ...

  10. 如何在Windows下用cpu模式跑通py-faster-rcnn 的demo.py

    关键字:Windows.cpu模式.Python.faster-rcnn.demo.py 声明:本篇blog暂时未经二次实践验证,主要以本人第一次配置过程的经验写成.计划在7月底回家去电脑城借台机子试 ...

随机推荐

  1. Java注解(3):一个真实Elasticsearch案例

    学会了技术就要使用,否则很容易忘记,因为自然界压根就不存在什么代码.变量之类的玩意,这都是一些和生活常识格格不入的东西.只能多用多练,形成肌肉记忆才行. 在一次实际的产品开发中,由于业务需求的缘故,需 ...

  2. 【JavaWeb】学习笔记——Servlet、Filter、Listenter

    Servlet Servlet 简介 Servlet 是 Java提供的一门动态web资源开发技术 Servlet 是JavaEE 规范之一,其实就是一个接口,将来我们需要定义Servlet类实现Se ...

  3. Android 跨进程渲染

    本项目用于验证 Android 是否能够跨进程渲染 View,最终实现了在子进程创建WebView,主进程显示的功能. 一.跨进程渲染的意义 有一些组件比如 WebView 如果在主进程初始化,会大大 ...

  4. Excel中的VLOOKUP函数

    VLOOKUP函数是Excel中的一个纵向查找函数,功能是按列查找,最终返回该列所需查询序列所对应的值. 该函数的语法规则如下: VLOOKUP(lookup_value,table_array,co ...

  5. 2022春每日一题:Day 35

    题目:[NOI Online #1 提高组] 冒泡排序 看到范围这么大,求逆序对,有修改,估计也只能树状数组了,考查冒泡排序性质,排第i次冒泡排序,总逆序对个数会减少i的逆序对个数,然后交换两个数,他 ...

  6. 并发bug之源(二)-有序性

    什么是有序性? 简单来说,假设你写了下面的程序: int a = 1; int b = 2; System.out.println(a); System.out.println(b); 但经过编译器/ ...

  7. 关于Go你不得不知道的小技巧

    目录 Go 箴言 Go 之禅 代码 使用 go fmt 格式化 多个 if 语句可以折叠成 switch 用 chan struct{} 来传递信号, chan bool 表达的不够清楚 30 * t ...

  8. 再见CMS

    观察网站最下方,根据备案号搜到这是个齐博CMS,然后百度就可以搜到齐博CMS漏洞了 然后开始利用 首先尝试了在用户信息修改处进行注入,发现好像想不通了,就在博客界面进行注入 Payload: 获取版本 ...

  9. 树莓派配置uwsgi服务

    前言 我配置 uwsgi 服务是为了运行给 python flask 项目,如果直接 pip3 install uwsgi 得到的uwsgi服务可以直接使用,只不过需要在命令行中启动服务(当然也可以使 ...

  10. 关于windows上开启远程桌面连接不上的问题解决

    解决办法 启用远程桌面连接(计算机-属性-远程设置) 选择第二个:允许运行任意版本远程桌面 关闭防火墙(或者给远程桌面添加端口3389放行也可以) 创建用户以及密码 这里注意,一定要建立密码,只有账号 ...