商城首页欢迎来到中国正版软件门户

您的位置:首页 >c++如何读取Linux系统的CPU负载信息_/proc/stat解析【实战】

c++如何读取Linux系统的CPU负载信息_/proc/stat解析【实战】

  发布于2026-05-03 阅读(0)

扫一扫,手机访问

实战解析:如何用C++精准读取Linux系统的CPU负载信息

c++如何读取Linux系统的CPU负载信息_/proc/stat解析【实战】

在性能监控和系统调优时,CPU使用率是一个绕不开的核心指标。很多开发者第一反应是去调用系统命令,但直接在程序中解析系统数据源,往往能获得更高效、更灵活的解决方案。今天,我们就来深入聊聊如何从/proc/stat这个宝藏文件中,用C++提取出我们需要的CPU负载信息。

直接读 /proc/stat 能拿到 CPU 负载,但不是“负载值”,是原始计时器累加值

这里有个关键概念需要先厘清:打开/proc/stat,第一行以cpu (注意末尾有个空格)开头,后面跟着的那一串数字,并不是我们通常理解的“实时百分比”或“1分钟平均负载”。后者其实是/proc/loada vg的职责。那么/proc/stat提供的是什么?它记录的是自系统启动以来,CPU在各种状态下所消耗的「jiffies」累计值。所以,想得到“最近一段时间CPU的忙碌占比”,你得自己动手:采样两次,计算差值,最后再做归一化处理。这个过程,本质上是在做一次微积分里的“求导”。

解析 /proc/stat 的前 4 列就能算出基础使用率(user + nice + system + idle)

在标准的内核输出里,cpu 行的前几个字段含义是相对固定的(后面的字段如iowaitirq等,则取决于内核版本和配置,不一定存在):

  • user:普通进程在用户态消耗的jiffies。
  • nice:低优先级(nice值大于0)进程在用户态消耗的jiffies。
  • system:内核态消耗的jiffies。
  • idle:CPU空闲时间(注意,这里不包括iowait)。

计算总时间,理论上可以把所有字段加起来,即 user + nice + system + idle + iowait + irq + softirq + steal。但更稳健的做法是使用std::accumulate遍历所有解析到的数字,避免因字段增减而导致硬编码下标越界。这里有个实操要点:两次读取的间隔时间不宜过短,建议至少200毫秒,否则差值太小,浮点运算的精度误差可能会被放大,导致结果失真。

用 C++ 读取并计算一次 CPU 使用率的最小可行代码

#include 
#include 
#include 
#include 
#include 
#include 

std::vector read_cpu_times() {
    std::ifstream f("/proc/stat");
    std::string line;
    while (std::getline(f, line)) {
        if (line.substr(0, 4) == "cpu ") { // 注意末尾空格
            std::vector res;
            std::stringstream ss(line.substr(4)); // 跳过 "cpu "
            long long val;
            while (ss >> val) res.push_back(val);
            return res;
        }
    }
    return {};
}

double calc_cpu_usage(const std::vector& a, const std::vector& b) {
    if (a.empty() || b.empty() || a.size() != b.size()) return -1.0;
    long long total_a = 0, total_b = 0, idle_a = 0, idle_b = 0;
    for (size_t i = 0; i < a.size(); ++i) {
        total_a += a[i];
        total_b += b[i];
        if (i == 3 && i < a.size()) { // 假设第 4 个是 idle(索引 3)
            idle_a = a[i];
            idle_b = b[i];
        }
    }
    long long total_delta = total_b - total_a;
    long long idle_delta = idle_b - idle_a;
    return (total_delta > 0) ? (1.0 - static_cast(idle_delta) / total_delta) * 100.0 : 0.0;
}

// 使用示例:
// auto a = read_cpu_times();
// std::this_thread::sleep_for(std::chrono::milliseconds(500));
// auto b = read_cpu_times();
// printf("CPU usage: %.1f%%\n", calc_cpu_usage(a, b));

常见坑:字段数量不固定、jiffies 单位模糊、多核汇总逻辑易错

把代码跑起来只是第一步,要想在生产环境中稳定可靠,还得避开几个“坑”。首先,/proc/stat的格式并非一成不变。内核版本升级可能会引入如guestguest_nice等新字段;而在某些嵌入式或定制内核中,字段也可能被删减。因此,依赖固定索引去取iowaitsteal值的做法非常危险。如果追求高精度和健壮性,更推荐的做法是按字段名进行匹配解析。

其次,关于jiffies的单位,它通常是USER_HZ(一般等于100),但好消息是,由于我们计算的是比例,单位在分子分母中会被约掉,所以用户空间的程序通常无需关心具体的换算。

最后,别忘了/proc/stat中的cpu 行是所有逻辑核心的统计数据之和。如果你想分析单个核心的负载,需要去读取cpu0cpu1等对应的独立行,并且要确保解析逻辑能处理这些行的字段顺序与汇总行一致的情况。

把这些细节都考虑到,你的CPU监控模块才算真正具备了实战能力。

立即学习“C++免费学习笔记(深入)”;

本文转载于:https://www.php.cn/faq/2318205.html 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。

热门关注