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)