windows和linux获取cpu占用率需系统api采样差值计算:windows用getprocesstimes与gettickcount64,linux读/proc/self/stat与/proc/stat,均需≥100ms间隔、两次采样求Δcpu时间/Δwall时间。

Windows 和 Linux 没有标准 C++ 接口能直接获取“当前进程 CPU 占用率”,必须调用系统 API,且原理不同:它本质是采样差值(% = ΔCPU 时间 / ΔWall 时间),不能单次调用就得出准确值。
Windows 下用 GetProcessTimes + 系统时间差计算
核心是对比两次采样间的内核态+用户态时间增量与真实流逝时间之比。注意:GetProcessTimes 返回的是 100ns 单位的 FILETIME,需转为毫秒或秒参与计算;必须至少间隔 100ms 以上采样才有效,否则分母太小、结果抖动极大。
- 第一次调用
GetProcessTimes后记录GetTickCount64()(推荐)或QueryPerformanceCounter() - 等待 ≥150ms(如
Sleep(200)),再调用一次GetProcessTimes - 计算:
(delta_kernel + delta_user) / (delta_wall_ms * 10000.0)→ 得到 0.0–1.0 区间的小数(即百分比 × 0.01) - 务必检查
GetProcessTimes返回值,失败时返回 0.0;GetCurrentProcess()句柄无需关闭
#include <windows.h>
#include <iostream>
double GetProcessCpuUsage() {
static FILETIME ftPrevSysKernel, ftPrevSysUser, ftPrevProcKernel, ftPrevProcUser;
static ULONGLONG prevTick = 0;
FILETIME ftSysKernel, ftSysUser, ftProcKernel, ftProcUser;
ULONGLONG curTick = GetTickCount64();
<pre class='brush:php;toolbar:false;'>if (!GetSystemTimes(&ftSysKernel, &ftSysUser, nullptr) ||
!GetProcessTimes(GetCurrentProcess(), &ftProcKernel, &ftProcUser, nullptr)) {
return 0.0;
}
if (prevTick == 0) {
ftPrevSysKernel = ftSysKernel; ftPrevSysUser = ftSysUser;
ftPrevProcKernel = ftProcKernel; ftPrevProcUser = ftProcUser;
prevTick = curTick;
return 0.0;
}
ULONGLONG sysKernel = ((ULONGLONG)ftSysKernel.dwHighDateTime << 32) + ftSysKernel.dwLowDateTime;
ULONGLONG sysUser = ((ULONGLONG)ftSysUser.dwHighDateTime << 32) + ftSysUser.dwLowDateTime;
ULONGLONG procKernel = ((ULONGLONG)ftProcKernel.dwHighDateTime << 32) + ftProcKernel.dwLowDateTime;
ULONGLONG procUser = ((ULONGLONG)ftProcUser.dwHighDateTime << 32) + ftProcUser.dwLowDateTime;
double deltaSys = (sysKernel + sysUser) - ((ULONGLONG)ftPrevSysKernel.dwHighDateTime << 32 + ftPrevSysKernel.dwLowDateTime +
(ULONGLONG)ftPrevSysUser.dwHighDateTime << 32 + ftPrevSysUser.dwLowDateTime);
double deltaProc = (procKernel + procUser) - ((ULONGLONG)ftPrevProcKernel.dwHighDateTime << 32 + ftPrevProcKernel.dwLowDateTime +
(ULONGLONG)ftPrevProcUser.dwHighDateTime << 32 + ftPrevProcUser.dwLowDateTime);
double deltaMs = curTick - prevTick;
ftPrevSysKernel = ftSysKernel; ftPrevSysUser = ftSysUser;
ftPrevProcKernel = ftProcKernel; ftPrevProcUser = ftProcUser;
prevTick = curTick;
if (deltaMs < 100) return 0.0; // 避免噪声
return (deltaProc / deltaSys) * 100.0; // 相对系统总 CPU 时间占比}
Linux 下读 /proc/self/stat 并解析字段
Linux 中每个进程的 /proc/[pid]/stat 第 14–17 字段分别是 utime、stime、cutime、cstime(单位:clock ticks),而系统总 jiffies 可从 /proc/stat 的第一行 cpu 行累加前 10 个数字得到。关键点:必须用 sysconf(_SC_CLK_TCK) 获取真实 tick 频率(通常是 100),不能硬编码;且两次采样间隔仍需 ≥100ms。
立即学习“C++免费学习笔记(深入)”;
- 读取
/proc/self/stat解析第 14–17 字段(注意字段索引从 1 开始,空格分割) - 读取
/proc/stat第一行,跳过 "cpu" 后累加前 10 个数字作为总 jiffies - 两次采样后,CPU 使用率 =
(Δprocess_jiffies / Δtotal_jiffies) × 100.0 - 字段易错:第 14 字段是 utime(用户态),15 是 stime(内核态),16/17 是子进程的,通常只加前两个
#include <fstream>
#include <sstream>
#include <vector>
#include <unistd.h>
#include <sys/sysconf.h>
double GetProcessCpuUsage() {
static long prevUtime = 0, prevStime = 0;
static unsigned long long prevTotalJiffies = 0;
static long clkTck = sysconf(_SC_CLK_TCK);
<pre class='brush:php;toolbar:false;'>std::ifstream stat("/proc/self/stat");
long utime = 0, stime = 0;
if (stat.is_open()) {
std::string line;
std::getline(stat, line);
std::istringstream iss(line);
std::vector<std::string> tokens;
std::string token;
while (iss >> token) tokens.push_back(token);
if (tokens.size() >= 17) {
utime = std::stol(tokens[13]); // index 13 → field 14
stime = std::stol(tokens[14]); // index 14 → field 15
}
}
std::ifstream procStat("/proc/stat");
unsigned long long totalJiffies = 0;
if (procStat.is_open()) {
std::string line;
std::getline(procStat, line);
std::istringstream iss(line);
std::string cpu;
iss >> cpu; // skip "cpu"
unsigned long val;
for (int i = 0; i < 10 && iss >> val; ++i) totalJiffies += val;
}
if (prevUtime == 0 || prevStime == 0) {
prevUtime = utime; prevStime = stime;
prevTotalJiffies = totalJiffies;
return 0.0;
}
long deltaProc = (utime + stime) - (prevUtime + prevStime);
unsigned long long deltaTotal = totalJiffies - prevTotalJiffies;
prevUtime = utime; prevStime = stime;
prevTotalJiffies = totalJiffies;
if (deltaTotal == 0) return 0.0;
return (static_cast<double>(deltaProc) / deltaTotal) * 100.0;}
跨平台封装要注意的陷阱
别试图用同一套逻辑在 Windows/Linux 上跑通——字段含义、时间单位、采样机制完全不同。最稳妥做法是条件编译,或者抽象出统一接口但内部完全隔离实现。
- Windows 下
GetProcessTimes对挂起进程仍返回有效值;Linux 下若进程刚启动,/proc/self/stat中 utime/stime 可能为 0,首次采样结果不可信 - 不要用
clock()或std::chrono替代系统级 wall time:Windows 要用GetTickCount64,Linux 用clock_gettime(CLOCK_MONOTONIC, ...) - CPU 占用率不是瞬时值,是窗口平均值;显示时建议做简单滑动平均(如保留最近 3 次结果取均值),避免界面跳变
- 权限问题:Linux 下普通进程读
/proc/[pid]/stat没限制,但读其他进程需同组或 root;Windows 下默认可查自身
真正难的不是调 API,而是理解“CPU 占用率”本身是个统计估算值——它依赖采样窗口、系统调度精度、以及你是否把子进程时间算进去。生产环境建议至少 200ms 以上采样间隔,并丢弃首两次结果。











