服务器之家:专注于服务器技术及软件下载分享
分类导航

Linux|Centos|Ubuntu|系统进程|Fedora|注册表|Bios|Solaris|Windows7|Windows10|Windows11|windows server|

服务器之家 - 服务器系统 - Linux - 聊聊Linux系统Top 命令中的 CPU 使用率

聊聊Linux系统Top 命令中的 CPU 使用率

2021-05-31 23:46Linux内核那些事songsong001 Linux

当Linux系统中没有可运行的进程时,将会执行 idle 进程。也就是说,当系统执行 idle 进程时,表示系统正处于空闲状态。

聊聊Linux系统Top 命令中的 CPU 使用率

平常我们使用 top 命令来查看系统的性能情况,在 top 命令中可以看到很多不同类型的 CPU 使用率,如下图红框中标出部分:

聊聊Linux系统Top 命令中的 CPU 使用率

下面,我们来介绍一下这些 CPU 使用率的意义:

  • us:user time,表示 CPU 执行用户进程的时间,包括 nice 时间。通常都是希望用户空间CPU越高越好。
  • sy:system time,表示 CPU 在内核运行的时间,包括 IRQ 和 softirq。系统 CPU 占用越高,表明系统某部分存在瓶颈。通常这个值越低越好。
  • ni:nice time,具有优先级的用户进程执行时占用的 CPU 利用率百分比。
  • id:idle time,表示系统处于空闲期,等待进程运行。
  • wa:waiting time,表示 CPU 在等待 IO 操作完成所花费的时间。系统不应该花费大量的时间来等待 IO 操作,否则就说明 IO 存在瓶颈。
  • hi:hard IRQ time,表示系统处理硬中断所花费的时间。
  • si:soft IRQ time,表示系统处理软中断所花费的时间。
  • st:steal time,被强制等待(involuntary wait)虚拟 CPU 的时间,此时 Hypervisor 在为另一个虚拟处理器服务。

当然,单靠上面的解释来理解它们的意义还是比较困难的。所以,本文主要从源码的角度来分析它们到底代表什么。

时钟中断

 

首先,我们要知道统计 CPU 使用情况在什么地方执行的。在分析之前,我们先来了解下 时钟中断:

时钟中断:是一种硬中断,由时间硬件(系统定时器,一种可编程硬件)产生。当 CPU 接收到时钟中断信号后,会在处理完当前指令后调用 时钟中断处理程序 来完成更新系统时间、执行周期性任务等。

可以发现,统计 CPU 使用情况是在 时钟中断处理程序 中完成的。

每个 CPU 的使用情况通过 cpu_usage_stat 结构来记录,我们来看看其定义:

  1. struct cpu_usage_stat { 
  2.     cputime64_t user
  3.     cputime64_t nice; 
  4.     cputime64_t system; 
  5.     cputime64_t softirq; 
  6.     cputime64_t irq; 
  7.     cputime64_t idle; 
  8.     cputime64_t iowait; 
  9.     cputime64_t steal; 
  10.     cputime64_t guest; 
  11. }; 

从 cpu_usage_stat 结构的定义可以看出,其每个字段与 top 命令的 CPU 使用率类型一一对应。在内核初始化时,会为每个 CPU 创建一个 cpu_usage_stat 结构,用于统计 CPU 的使用情况。

OK,现在我们来分析下内核是怎么统计 CPU 的使用情况的。

每次执行 时钟中断处理程序 都会调用 account_process_tick 函数进行 CPU 使用情况统计,我们来分析一下 account_process_tick 函数的实现:

  1. void account_process_tick(struct task_struct *p, int user_tick) 
  2.     cputime_t one_jiffy_scaled = cputime_to_scaled(cputime_one_jiffy); 
  3.     struct rq *rq = this_rq(); 
  4.  
  5.     // 说明:user_tick 变量标识当前是否处于执行用户应用程序 
  6.  
  7.     if (user_tick) { 
  8.         // 1. 如果 CPU 在执行用户程序, 那么调用 account_user_time 进行统计 
  9.         account_user_time(p, cputime_one_jiffy, one_jiffy_scaled); 
  10.     } else if ((p != rq->idle) || (irq_count() != HARDIRQ_OFFSET)) { 
  11.         // 2. 如果 CPU 在执行内核代码, 那么调用 account_system_time 进行统计 
  12.         account_system_time(p, HARDIRQ_OFFSET, cputime_one_jiffy, 
  13.                             one_jiffy_scaled); 
  14.     } else { 
  15.         // 3. 否则说明 CPU 在执行 idle 进程(也就是处于空闲状态), 那么调用 account_idle_time 进行统计 
  16.         account_idle_time(cputime_one_jiffy); 
  17.     } 

account_process_tick 函数主要分 3 种情况进行统计,如下:

如果 CPU 在执行用户程序,那么调用 account_user_time 进行统计。

如果 CPU 在执行内核代码,那么调用 account_system_time 进行统计。

否则说明 CPU 在执行 idle 进程(也就是处于空闲状态),那么调用 account_idle_time 进行统计。

CPU 使用情况统计

 

下面我们分别对这 3 种统计进行分析。

1. 统计用户程序执行时间

统计用户程序的执行时间是通过 account_user_time 函数来完成的,我们来看看其实现:

  1. void account_user_time(struct task_struct *p, cputime_t cputime, 
  2.                        cputime_t cputime_scaled) 
  3.     // 获取 CPU 的统计结构(每个CPU一个 cpu_usage_stat 结构) 
  4.     struct cpu_usage_stat *cpustat = &kstat_this_cpu.cpustat;  
  5.     cputime64_t tmp; 
  6.     ... 
  7.  
  8.     // 分 2 种情况统计 CPU 的使用情况 
  9.     // 1. 如果进程的 nice 值大于0, 那么将会统计到 nice 字段中 
  10.     // 2. 如果进程的 nice 值小于等于0, 那么将会统计到 user 字段中 
  11.     if (TASK_NICE(p) > 0) 
  12.         cpustat->nice = cputime64_add(cpustat->nice, tmp); 
  13.     else 
  14.         cpustat->user = cputime64_add(cpustat->user, tmp); 
  15.     ... 

account_user_time 函数主要分两种情况统计:

  • 如果进程的 nice 值大于0,那么将会增加到 CPU 统计结构的 nice 字段中。
  • 如果进程的 nice 值小于等于0,那么增加到 CPU 统计结构的 user 字段中。

这里说明一下进程 nice 值的作用,nice 值越大,说明进程的优先级越低。所以,nice 统计值主要用来统计低优先级进程的占使用 CPU 的情况。也说明了,user 和 nice 统计值都属于执行用户程序的 CPU 时间。

2. 统计内核代码执行时间

如果在发生时钟中断前,CPU 处于内核态,也就是说在执行内核代码。那么将会调用 account_system_time 函数进行统计,account_system_time 函数实现如下:

  1. void account_system_time(struct task_struct *p, int hardirq_offset, 
  2.                          cputime_t cputime, cputime_t cputime_scaled) 
  3.     // 获取 CPU 的统计结构(每个CPU一个 cpu_usage_stat 结构) 
  4.     struct cpu_usage_stat *cpustat = &kstat_this_cpu.cpustat; 
  5.     cputime64_t tmp; 
  6.     ... 
  7.  
  8.     // 主要分 3 种情况进行统计 
  9.     // 1. 如果当前处于硬中断执行上下文, 那么统计到 irq 字段中 
  10.     // 2. 如果当前处于软中断执行上下文, 那么统计到 softirq 字段中 
  11.     // 3. 否则统计到 system 字段中 
  12.     if (hardirq_count() - hardirq_offset) 
  13.         cpustat->irq = cputime64_add(cpustat->irq, tmp); 
  14.     else if (softirq_count()) 
  15.         cpustat->softirq = cputime64_add(cpustat->softirq, tmp); 
  16.     else 
  17.         cpustat->system = cputime64_add(cpustat->system, tmp); 
  18.     ... 

account_system_time 函数主要分 3 种情况进行统计:

如果当前处于硬中断执行上下文,那么增加到 CPU 统计结构的 irq 字段中。

如果当前处于软中断执行上下文,那么增加到 CPU 统计结构的 softirq 字段中。

否则增加到 CPU 统计结构的 system 字段中。

从上面代码可以看出,irq 和 softirq 统计值也算是内核代码执行时间。

3. idle 进程执行时间统计

当系统中没有可运行的进程时,将会执行 idle 进程。也就是说,当系统执行 idle 进程时,表示系统正处于空闲状态。

idle 进程执行时间统计由 account_idle_time 函数完成,其实现如下:

  1. void account_idle_time(cputime_t cputime) 
  2.     struct cpu_usage_stat *cpustat = &kstat_this_cpu.cpustat; 
  3.     cputime64_t cputime64 = cputime_to_cputime64(cputime); 
  4.     struct rq *rq = this_rq(); 
  5.  
  6.     // 分 2 种情况统计 CPU 的使用情况 
  7.     // 1. 如果系统有进程正在等待 I/O 操作完成, 那么将统计到 iowait 字段中 
  8.     // 2. 否则将统计到 idle 字段中 
  9.     if (atomic_read(&rq->nr_iowait) > 0) 
  10.         cpustat->iowait = cputime64_add(cpustat->iowait, cputime64); 
  11.     else 
  12.         cpustat->idle = cputime64_add(cpustat->idle, cputime64); 

account_idle_time 函数也分两种情况进行统计:

  • 如果系统中有正在等待 I/O 操作完成的进程,那么增加到 CPU 统计结构的 iowait 字段中。
  • 否则增加到 CPU 统计结构的 idle 字段中。

从上面的分析可以看出,iowait 统计值也属于空闲时间的一种。

top 命令的 CPU 使用率

 

通过源码分析,我们知道 top 命令中 CPU 使用率各种类型的意思,现在我们来介绍一下 top 命令是怎么计算各种类型的 CPU 使用率。

要获取各个 CPU 的使用情况信息,可以通过读取 /proc/stat 文件获取,如下:

  1. [vagrant@localhost ~]$ cat /proc/stat 
  2. cpu  245 10 1142 1097923 95 0 28 0 0 0 
  3. cpu0 245 10 1142 1097923 95 0 28 0 0 0 
  4. ... 

上面的结果显示了 CPU 的使用情况信息,第一行代表所有 CPU 的总和,而第二行开始表示每个 CPU 核心的使用情况信息。因为我的电脑只有一个核,所以只有一条数据。

下面说说这些数据的意义,从第一个数值开始分别代表:user ,nice,system,idle,iowait, irq,softirq,steal。

所以,top 命令的 CPU 使用率计算公式如下:

  1. CPU总时间 = user + nice + system + idle + wait + irq + softirq + steal 
  2. %us = user / CPU总时间 
  3. %ni = nice / CPU总时间 
  4. %sy = system / CPU总时间 
  5. %id = idel / CPU总时间 
  6. %wa = wait / CPU总时间 
  7. %hi = irq / CPU总时间 
  8. %si = softirq / CPU总时间 
  9. %st = steal / CPU总时间 

嗯,看起来还是挺简单的。

总结

 

本文主要分析了 top 命令中的 CPU 使用率的意义和实现原理,希望通过本文,能够帮助大家对 top 命令有更深的认识。

原文链接:https://mp.weixin.qq.com/s/qkjGYoheHvs-lX9avrYg_g

延伸 · 阅读

精彩推荐
  • LinuxLinux lnmp下无法使用mail发邮件的两种解决方法

    Linux lnmp下无法使用mail发邮件的两种解决方法

    在配置了lnmp环境后,出现了mail函数不能发送邮件的问题,其实有两种方法,一是使用sendmail组件,而是使用postfix。下面为大家一一介绍下 ...

    Linux之家4042019-09-17
  • Linux手把手教您在 Linux 上使用 GPG 加解密文件

    手把手教您在 Linux 上使用 GPG 加解密文件

    在本教程中,我将告诉你如何用 GPG 加密和解密文件。这是一个简单的教程,你可以在你的 Linux 系统上尝试所有的练习。这将帮助你练习 GPG 命令,并在你...

    Linux中国6962021-12-15
  • LinuxLinux上设置用户通过SFTP访问目录的权限的方法

    Linux上设置用户通过SFTP访问目录的权限的方法

    这篇文章主要介绍了Linux上设置用户通过SFTP访问目录的权限的方法,SFTP可以理解为使用SSH协议进行FTP传输的协议,因而同时要对OpenSSH进行相关设置,需要的朋...

    OSChina10022019-06-19
  • Linuxlinux中rmdir命令使用详解(删除空目录)

    linux中rmdir命令使用详解(删除空目录)

    今天学习一下linux中命令: rmdir命令。rmdir是常用的命令,该命令的功能是删除空目录,一个目录被删除之前必须是空的 ...

    linux命令大全5372019-11-19
  • LinuxLinux常用的日志文件和常用命令

    Linux常用的日志文件和常用命令

    成功地管理任何系统的关键之一,是要知道系统中正在发生什么事。 Linux 中提供了异常日志,并且日志的细节是可配置的。Linux 日志都以明文形式存储,所...

    Linux教程网2632020-04-18
  • Linux确保Linux系统安全的前提条件 漏洞防护

    确保Linux系统安全的前提条件 漏洞防护

    Linux 作为开放式的操作系统受到很多程序员的喜爱,很多高级程序员都喜欢编写Linux操作系统的相关软件。这使得Linux操作系统有着丰富的软件支持,还有无...

    Linux之家2642020-04-11
  • LinuxLinux中环境变量配置的步骤详解

    Linux中环境变量配置的步骤详解

    Linux中环境变量包括系统级和用户级,系统级的环境变量是每个登录到系统的用户都要读取的系统变量,而用户级的环境变量则是该用户使用系统时加载的...

    Myths7882022-02-10
  • Linux详解Linux系统下PXE服务器的部署过程

    详解Linux系统下PXE服务器的部署过程

    这篇文章主要介绍了Linux系统下PXE服务器的部署过程,包括对PXE的API架构作了一个基本的简介,需要的朋友可以参考下...

    运维之道9812019-07-04