Python监控CPU内存使用率方法
时间:2026-05-22 23:06:34 357浏览 收藏
本文深入解析了使用 Python 的 psutil 库监控系统 CPU 和内存使用率时常见却易被误解的核心问题:为什么 `psutil.cpu_percent()` 首次调用总返回 0.0(实为依赖时间差采样的正常设计,需预热)、如何正确获取真实可用内存(应依赖 `available` 而非 `used`,避免误判内存压力)、在容器或低权限环境下规避 `PermissionError` 的降级策略、长期运行脚本内存泄漏的根源(进程缓存累积)及 `psutil.cleanup()` 的关键作用——揭示了 psutil 并非无状态工具函数,而是一个有采样逻辑、权限边界和内部状态的系统级监控组件,掌握这些“反直觉”细节,才能写出稳定、准确、生产就绪的资源监控代码。

psutil.cpu_percent() 为什么第一次调用总是返回 0.0?
因为 psutil.cpu_percent() 是基于时间差的采样函数,首次调用时没有前一次基准值,只能返回 0.0。这不是 bug,是设计使然。
实际使用必须先调用一次并等待至少 0.1 秒以上,再调用才得到有效值:
import psutil import time <p>psutil.cpu_percent() # 首次“预热”,丢弃 time.sleep(0.1) print(psutil.cpu_percent()) # 正确值,例如 12.3</p>
- 如果想每秒刷新一次,
interval=1参数可省略 sleep,但首次仍需预热或设percpu=False+interval=1直接阻塞等待 - 多核场景下,
psutil.cpu_percent(percpu=True)返回列表,每个元素对应一个逻辑 CPU 的使用率 - 在容器环境(如 Docker)中,该值反映的是宿主机整体 CPU 负载,不是容器 cgroup 限制后的实际使用率
psutil.virtual_memory() 返回的 total、used、available 哪个才代表“真正可用内存”?
available 是唯一能安全用于判断系统是否吃紧的字段;used 包含缓存和 buffers,会误导你认为内存快爆了。
Linux 下内核会把空闲内存用于 page cache,这部分在需要时可立即回收,所以 available = free + 可快速回收的 cache/buffers。
- 不要用
used / total > 0.9做告警,应该用(total - available) / total > 0.9 psutil.virtual_memory().percent内部就是基于available计算的,等价于(total - available) / total * 100- Windows 上
available含义略有不同(包含 standby list),但仍是比used更可靠的指标
如何避免 psutil 在低权限环境(如容器、非 root 用户)下报 PermissionError?
常见错误信息:PermissionError: [Errno 13] Permission denied: '/proc/1/status' 或读取某些进程信息失败。
psutil 默认尝试获取所有进程的详细信息,但在受限环境中会被拒绝。解决思路是降级容忍——只取全局统计,跳过敏感进程。
- 用
psutil.cpu_percent()和psutil.virtual_memory()安全,它们不依赖单个进程权限 - 若还需进程级监控(比如 top 10 占用),改用
psutil.process_iter(['pid', 'name', 'cpu_percent', 'memory_info'])并捕获异常:
for proc in psutil.process_iter(['pid', 'name', 'memory_info']):
try:
print(proc.info['pid'], proc.info['name'], proc.info['memory_info'].rss)
except (psutil.NoSuchProcess, psutil.AccessDenied, psutil.ZombieProcess):
pass- Docker 容器中建议挂载
/proc:/proc:ro,否则psutil无法读取关键路径
监控脚本跑久了内存持续上涨?可能是 psutil 缓存未清理
psutil 内部对进程对象做了弱引用缓存,长时间运行且频繁调用 psutil.process_iter() 会导致缓存累积,尤其在进程数多变的环境中。
官方推荐做法:显式调用 psutil.cleanup() 清理缓存(v5.9.0+),或控制迭代范围避免无谓遍历。
- 每轮监控后加
psutil.cleanup(),尤其在循环中反复调用process_iter()时 - 避免写
list(psutil.process_iter()),这会一次性加载全部进程对象到内存;改用迭代器 + 条件 break - 若只关心系统级指标(CPU/内存),根本不需要调用
process_iter(),也就无需担心缓存问题
psutil 的坑大多出在“以为它像普通函数一样调了就返回,其实它有状态、有采样依赖、有权限边界”。最稳妥的做法是:只拿你需要的字段,提前处理异常,别让它碰权限死角。
今天带大家了解了的相关知识,希望对你有所帮助;关于文章的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
161 收藏
-
347 收藏
-
241 收藏
-
357 收藏
-
202 收藏
-
259 收藏
-
347 收藏
-
400 收藏
-
370 收藏
-
245 收藏
-
371 收藏
-
135 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习