c++如何获取当前进程CPU占用率_c++ Windows/Linux系统API调用【实战】
发布时间 - 2026-01-06 00:00:00 点击率:次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#include double GetProcessCpuUsage() { static FILETIME ftPrevSysKernel, ftPrevSysUser, ftPrevProcKernel, ftPrevProcUser; static ULONGLONG prevTick = 0; FILETIME ftSysKernel, ftSysUser, ftProcKernel, ftProcUser; ULONGLONG curTick = GetTickCount64(); 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 zuojiankuohaophpcnzuojiankuohaophpcn 32) + ftSysKernel.dwLowDateTime; ULONGLONG sysUser = ((ULONGLONG)ftSysUser.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32) + ftSysUser.dwLowDateTime; ULONGLONG procKernel = ((ULONGLONG)ftProcKernel.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32) + ftProcKernel.dwLowDateTime; ULONGLONG procUser = ((ULONGLONG)ftProcUser.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32) + ftProcUser.dwLowDateTime; double deltaSys = (sysKernel + sysUser) - ((ULONGLONG)ftPrevSysKernel.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32 + ftPrevSysKernel.dwLowDateTime + (ULONGLONG)ftPrevSysUser.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32 + ftPrevSysUser.dwLowDateTime); double deltaProc = (procKernel + procUser) - ((ULONGLONG)ftPrevProcKernel.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32 + ftPrevProcKernel.dwLowDateTime + (ULONGLONG)ftPrevProcUser.dwHighDateTime zuojiankuohaophpcnzuojiankuohaophpcn 32 + ftPrevProcUser.dwLowDateTime); double deltaMs = curTick - prevTick; ftPrevSysKernel = ftSysKernel; ftPrevSysUser = ftSysUser; ftPrevProcKernel = ftProcKernel; ftPrevProcUser = ftProcUser; prevTick = curTick; if (deltaMs zuojiankuohaophpcn 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。
- 读取
/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#include #include #include #include double GetProcessCpuUsage() { static long prevUtime = 0, prevStime = 0; static unsigned long long prevTotalJiffies = 0; static long clkTck = sysconf(_SC_CLK_TCK); 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::vectorzuojiankuohaophpcnstd::string> tokens; std::string token; while (iss youjiankuohaophpcnyoujiankuohaophpcn token) tokens.push_back(token); if (tokens.size() youjiankuohaophpcn= 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 youjiankuohaophpcnyoujiankuohaophpcn cpu; // skip "cpu" unsigned long val; for (int i = 0; i zuojiankuohaophpcn 10 && iss youjiankuohaophpcnyoujiankuohaophpcn 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_castzuojiankuohaophpcndouble>(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 以上采样间隔,并丢弃首两次结果。
# linux # windows # 编码 # c++ # ios # win # stream # linux系统 # api调用 # 封装 # 接口 # 两次 # 占用率 # 的是 # 有效值 # 是个 # 首次 # 句柄 # 要注意 # 要用 # 你是否
相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571 】
相关推荐: Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理 如何自己制作一个网站链接,如何制作一个企业网站,建设网站的基本步骤有哪些? 打开php文件提示内存不足_怎么调整php内存限制【解决方案】 Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤 Windows家庭版如何开启组策略(gpedit.msc)?(安装方法) 如何在建站主机中优化服务器配置? html5audio标签播放结束怎么触发事件_onended回调方法【教程】 Windows10如何删除恢复分区_Win10 Diskpart命令强制删除分区 如何在香港服务器上快速搭建免备案网站? Laravel Session怎么存储_Laravel Session驱动配置详解 标题:Vue + Vuex + JWT 身份认证的正确实践与常见误区解析 今日头条微视频如何找选题 今日头条微视频找选题技巧【指南】 php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】 香港服务器如何优化才能显著提升网站加载速度? 微信h5制作网站有哪些,免费微信H5页面制作工具? 如何快速生成凡客建站的专业级图册? 如何在万网自助建站平台快速创建网站? Laravel如何使用Seeder填充数据_Laravel模型工厂Factory批量生成测试数据【方法】 Laravel如何优化应用性能?(缓存和优化命令) node.js报错:Cannot find module 'ejs'的解决办法 安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出 Linux网络带宽限制_tc配置实践解析【教程】 Java解压缩zip - 解压缩多个文件或文件夹实例 网站建设保证美观性,需要考虑的几点问题! 微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】 如何快速查询网址的建站时间与历史轨迹? 如何用低价快速搭建高质量网站? EditPlus中的正则表达式实战(6) Laravel请求验证怎么写_Laravel Validator自定义表单验证规则教程 中山网站制作网页,中山新生登记系统登记流程? ,南京靠谱的征婚网站? 如何在万网主机上快速搭建网站? 移动端脚本框架Hammer.js 再谈Python中的字符串与字符编码(推荐) 高防网站服务器:DDoS防御与BGP线路的AI智能防护方案 郑州企业网站制作公司,郑州招聘网站有哪些? 如何快速完成中国万网建站详细流程? Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】 如何用JavaScript实现文本编辑器_光标和选区怎么处理 网站页面设计需要考虑到这些问题 html5的keygen标签为什么废弃_替代方案说明【解答】 如何在不使用负向后查找的情况下匹配特定条件前的换行符 如何用好域名打造高点击率的自主建站? Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】 html5如何设置样式_HTML5样式设置方法与CSS应用技巧【教程】 微信小程序 五星评分(包括半颗星评分)实例代码 成都品牌网站制作公司,成都营业执照年报网上怎么办理? INTERNET浏览器怎样恢复关闭标签页_INTERNET浏览器标签恢复快捷键与方法【指南】 详解CentOS6.5 安装 MySQL5.1.71的方法 关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)


stream iss(line);
std::string cpu;
iss youjiankuohaophpcnyoujiankuohaophpcn cpu; // skip "cpu"
unsigned long val;
for (int i = 0; i zuojiankuohaophpcn 10 && iss youjiankuohaophpcnyoujiankuohaophpcn 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_castzuojiankuohaophpcndouble>(deltaProc) / deltaTotal) * 100.0;