0

0

使用Go语言在Linux系统下获取CPU使用率的教程

心靈之曲

心靈之曲

发布时间:2025-09-10 20:50:01

|

241人浏览过

|

来源于php中文网

原创

使用Go语言在Linux系统下获取CPU使用率的教程

本文详细介绍了如何在Go语言程序中,利用goprocinfo库在Linux系统下获取并计算CPU使用率。通过解析/proc/stat文件,我们可以获取系统和用户进程的CPU时间统计数据,并进一步计算出实时的CPU使用百分比。教程涵盖了库的安装、基本用法、核心计算逻辑及完整的代码示例,并提供了注意事项和最佳实践,帮助开发者高效监控系统CPU性能。

引言:Go程序中监控CPU使用率的重要性

在构建高性能和高可靠性的go应用程序时,实时监控系统资源,特别是cpu使用率,是不可或缺的一环。了解cpu的负载情况有助于我们诊断性能瓶颈、优化资源分配,并确保应用程序的稳定运行。在linux系统中,/proc/stat文件是获取cpu统计信息的标准接口。然而,直接解析这个文件可能比较繁琐。本教程将介绍如何利用go语言的goprocinfo库,便捷地获取并计算cpu使用率。

使用goprocinfo库获取CPU统计信息

goprocinfo是一个Go语言库,旨在简化对Linux /proc文件系统中信息的解析。它提供了一系列结构体和函数,用于读取和解析/proc/stat、/proc/meminfo等文件,从而方便地获取系统资源数据。

1. 安装goprocinfo库

首先,您需要将goprocinfo库添加到您的Go项目中。在您的终端中运行以下命令:

go get github.com/c9s/goprocinfo/linux

2. 基本使用:读取/proc/stat

goprocinfo库的linux子包提供了ReadStat()函数,用于读取并解析/proc/stat文件。该函数返回一个Stat结构体,其中包含了系统的CPU统计数据。

Stat结构体中的CPUStats字段是一个切片,它包含了每个CPU核心(以及一个总计CPU)的详细统计信息。每个元素都是一个CPUStat结构体,其字段代表了不同模式下CPU的累计时间(单位为jiffies,即内核时钟滴答数)。

立即学习go语言免费学习笔记(深入)”;

package main

import (
    "fmt"
    "log"
    "time"

    "github.com/c9s/goprocinfo/linux"
)

// CPUStatsSnapshot 结构体用于存储CPU的快照数据
type CPUStatsSnapshot struct {
    Total  uint64
    Active uint64
}

// getCPUStatsSnapshot 获取当前CPU的总时间与活跃时间快照
func getCPUStatsSnapshot() (CPUStatsSnapshot, error) {
    stat, err := linux.ReadStat("/proc/stat")
    if err != nil {
        return CPUStatsSnapshot{}, fmt.Errorf("读取/proc/stat失败: %w", err)
    }

    // 第一个CPUStats元素通常是所有CPU的总和
    if len(stat.CPUStats) == 0 {
        return CPUStatsSnapshot{}, fmt.Errorf("未找到CPU统计数据")
    }

    // 获取总CPU的统计数据
    cpu := stat.CPUStats[0] // 通常第一个是所有CPU的总和

    total := cpu.User + cpu.Nice + cpu.System + cpu.Idle + cpu.IOWait + cpu.IRQ + cpu.SoftIRQ + cpu.Steal + cpu.Guest + cpu.GuestNice
    active := cpu.User + cpu.Nice + cpu.System + cpu.IRQ + cpu.SoftIRQ + cpu.Steal + cpu.Guest + cpu.GuestNice

    return CPUStatsSnapshot{
        Total:  total,
        Active: active,
    }, nil
}

func main() {
    fmt.Println("正在获取CPU统计信息...")

    // 演示如何直接读取和打印原始数据
    stat, err := linux.ReadStat("/proc/stat")
    if err != nil {
        log.Fatalf("读取/proc/stat失败: %v", err)
    }

    fmt.Println("\n--- 原始CPU统计数据 ---")
    for i, s := range stat.CPUStats {
        cpuLabel := fmt.Sprintf("CPU%d", i)
        if i == 0 {
            cpuLabel = "总CPU"
        }
        fmt.Printf("%s: User=%d, Nice=%d, System=%d, Idle=%d, IOWait=%d, IRQ=%d, SoftIRQ=%d, Steal=%d, Guest=%d, GuestNice=%d\n",
            cpuLabel, s.User, s.Nice, s.System, s.Idle, s.IOWait, s.IRQ, s.SoftIRQ, s.Steal, s.Guest, s.GuestNice)
    }
}

在上面的代码中,我们展示了如何读取/proc/stat并遍历CPUStats切片。CPUStat结构体中的字段含义如下:

TGroupon分销系统
TGroupon分销系统

TGROUPON分销系统,隶属于易推软件所属旗下产品,TGROUPON简称TG分销系统。核心框架采用了ECSHOP+ECTOUCH系统,并使用PHP做为核心编程语言。TG分销系统前身为ThinkGroupon,始创于2011年,并拥有多项国家著作权、商标权。优势1:TG分销系统可以很好的融入到微信端,快速、便捷地接入微信公众号。轻轻松松获取微信端粉丝头像、姓名等一系列会员信息,有效的形成大数据数据

下载
  • User: 用户模式下CPU运行的时间。
  • Nice: 低优先级用户模式下CPU运行的时间。
  • System: 内核模式下CPU运行的时间。
  • Idle: CPU空闲时间。
  • IOWait: CPU等待I/O完成的时间。
  • IRQ: 处理硬件中断的时间。
  • SoftIRQ: 处理软件中断的时间。
  • Steal: 在虚拟化环境中,被其他虚拟机“偷走”的CPU时间。
  • Guest: 运行虚拟CPU的时间(作为客户机)。
  • GuestNice: 运行低优先级虚拟CPU的时间。

计算CPU使用率百分比

直接从/proc/stat获取的这些值是自系统启动以来的累计时间。要计算实时的CPU使用率百分比,我们需要在两个不同的时间点获取两次快照,然后计算这些值在时间间隔内的变化量。

CPU使用率的计算公式为: CPU使用率 = (两次采样间隔内活跃CPU时间增量 / 两次采样间隔内总CPU时间增量) * 100%

其中:

  • 活跃CPU时间增量 = (User_2 + Nice_2 + System_2 + IRQ_2 + SoftIRQ_2 + Steal_2 + Guest_2 + GuestNice_2) - (User_1 + Nice_1 + System_1 + IRQ_1 + SoftIRQ_1 + Steal_1 + Guest_1 + GuestNice_1)
  • 总CPU时间增量 = (User_2 + Nice_2 + System_2 + Idle_2 + IOWait_2 + IRQ_2 + SoftIRQ_2 + Steal_2 + Guest_2 + GuestNice_2) - (User_1 + Nice_1 + System_1 + Idle_1 + IOWait_1 + IRQ_1 + SoftIRQ_1 + Steal_1 + Guest_1 + GuestNice_1)

以下是一个完整的Go语言代码示例,演示如何计算系统总CPU使用率:

package main

import (
    "fmt"
    "log"
    "time"

    "github.com/c9s/goprocinfo/linux"
)

// CPUStatsSnapshot 结构体用于存储CPU的快照数据
type CPUStatsSnapshot struct {
    Total  uint64
    Active uint64
}

// getCPUStatsSnapshot 获取当前CPU的总时间与活跃时间快照
func getCPUStatsSnapshot() (CPUStatsSnapshot, error) {
    stat, err := linux.ReadStat("/proc/stat")
    if err != nil {
        return CPUStatsSnapshot{}, fmt.Errorf("读取/proc/stat失败: %w", err)
    }

    // 第一个CPUStats元素通常是所有CPU的总和
    if len(stat.CPUStats) == 0 {
        return CPUStatsSnapshot{}, fmt.Errorf("未找到CPU统计数据")
    }

    // 获取总CPU的统计数据
    cpu := stat.CPUStats[0] // 通常第一个是所有CPU的总和

    total := cpu.User + cpu.Nice + cpu.System + cpu.Idle + cpu.IOWait + cpu.IRQ + cpu.SoftIRQ + cpu.Steal + cpu.Guest + cpu.GuestNice
    active := cpu.User + cpu.Nice + cpu.System + cpu.IRQ + cpu.SoftIRQ + cpu.Steal + cpu.Guest + cpu.GuestNice

    return CPUStatsSnapshot{
        Total:  total,
        Active: active,
    }, nil
}

// CalculateCPULoad 计算CPU负载百分比
func CalculateCPULoad(prev, curr CPUStatsSnapshot) float64 {
    deltaTotal := float64(curr.Total - prev.Total)
    deltaActive := float64(curr.Active - prev.Active)

    if deltaTotal == 0 {
        return 0.0 // 避免除以零
    }

    return (deltaActive / deltaTotal) * 100.0
}

func main() {
    fmt.Println("正在监控CPU使用率...")

    // 第一次采样
    prevStats, err := getCPUStatsSnapshot()
    if err != nil {
        log.Fatalf("获取初始CPU快照失败: %v", err)
    }

    // 等待一段时间进行第二次采样
    interval := 3 * time.Second
    fmt.Printf("等待 %s 进行下一次采样...\n", interval)
    time.Sleep(interval)

    // 第二次采样
    currStats, err := getCPUStatsSnapshot()
    if err != nil {
        log.Fatalf("获取当前CPU快照失败: %v", err)
    }

    // 计算CPU使用率
    cpuUsage := CalculateCPULoad(prevStats, currStats)

    fmt.Printf("过去 %s 的CPU使用率: %.2f%%\n", interval, cpuUsage)

    // 可以在循环中持续监控
    fmt.Println("\n--- 持续监控 (按 Ctrl+C 退出) ---")
    prevStats = currStats // 将当前快照设为下一次的“前一次”
    ticker := time.NewTicker(interval)
    defer ticker.Stop()

    for range ticker.C {
        currStats, err := getCPUStatsSnapshot()
        if err != nil {
            log.Printf("获取CPU快照失败: %v", err)
            continue
        }

        cpuUsage = CalculateCPULoad(prevStats, currStats)
        fmt.Printf("过去 %s 的CPU使用率: %.2f%%\n", interval, cpuUsage)
        prevStats = currStats
    }
}

注意事项与最佳实践

  1. 错误处理:在实际应用中,务必对linux.ReadStat()等可能返回错误的操作进行适当的错误处理,以增强程序的健壮性。
  2. 采样间隔:选择合适的采样间隔非常重要。间隔太短可能导致测量不准确(例如,如果CPU在极短时间内爆发性使用),间隔太长则可能无法及时反映CPU负载的变化。通常,1到5秒的间隔是比较常见的选择。
  3. 总CPU与单核CPU:/proc/stat的第一个CPUStat条目(stat.CPUStats[0])通常代表所有CPU核心的总和。随后的条目(stat.CPUStats[1]、stat.CPUStats[2]等)则代表每个单独的CPU核心。根据您的需求,可以选择监控总CPU或特定核心的CPU使用率。
  4. 平台限制:goprocinfo库主要用于解析Linux系统的/proc文件。因此,此方法仅适用于Linux环境。如果您的应用程序需要在其他操作系统(如Windows或macOS)上运行,您需要使用针对这些平台的特定API或库来获取CPU信息。
  5. 单位jiffies:/proc/stat中所有时间相关的统计数据都以jiffies为单位,这是一个系统定义的时钟滴答数。虽然在计算百分比时,jiffies的具体值会被抵消,但了解其单位有助于理解原始数据。
  6. 资源消耗:频繁地读取/proc/stat并进行计算可能会消耗一定的CPU资源。在设计监控系统时,应权衡监控精度和资源消耗。

总结

通过goprocinfo库,Go语言开发者可以轻松地在Linux系统下获取详细的CPU统计信息。结合两次采样计算差值的原理,我们能够准确地计算出实时的CPU使用率百分比。这种方法为Go应用程序提供了强大的系统性能监控能力,有助于构建更稳定、更高效的服务。在实际部署时,请务必考虑错误处理、采样间隔和平台兼容性等因素,以确保监控系统的可靠性和准确性。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

197

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

189

2025.07.04

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1027

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

66

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

455

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

11

2026.01.19

Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

234

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

446

2023.09.25

excel表格操作技巧大全 表格制作excel教程
excel表格操作技巧大全 表格制作excel教程

Excel表格操作的核心技巧在于 熟练使用快捷键、数据处理函数及视图工具,如Ctrl+C/V(复制粘贴)、Alt+=(自动求和)、条件格式、数据验证及数据透视表。掌握这些可大幅提升数据分析与办公效率,实现快速录入、查找、筛选和汇总。

0

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.5万人学习

Git 教程
Git 教程

共21课时 | 2.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号