Python 怎么监控内存使用并在泄漏时自动报警?

发布时间 - 2026-01-23 00:00:00    点击率:
用psutil.Process(os.getpid()).memory_info().rss可准确获取Python进程真实内存占用(RSS),避免sys.getsizeof()等仅统计Python对象的局限;需每1–5秒采样,结合斜率与GC回落判断泄漏,多进程须独立监控,C扩展内存需依赖RSS趋势识别。

怎么用 psutil 实时查 Python 进程内存占用

直接靠 sys.getsizeof()gc.get_objects() 无法反映真实内存压力,它们只统计 Python 对象堆内大小,不包含 C 扩展、mmap、共享库等。真正可靠的是从操作系统层面读进程 RSS(Resident Set Size)。

psutil 是最轻量且跨平台的选择:

import psutil
import os

p = psutil.Process(os.getpid()) rss_mb = p.memory_info().rss / 1024 / 1024 # 转 MB print(f"当前 RSS: {rss_mb:.1f} MB")

  • 务必用 p.memory_info().rss,不是 vms(虚拟内存不可靠)
  • Linux 下 RSS 基本等于“实际占物理内存”,Windows/macOS 行为略有差异但足够用于趋势判断
  • 不要在循环里频繁调用——psutil.Process() 初始化有开销,建议每 1–5 秒采样一次

怎么定义“内存泄漏”并触发报警

内存泄漏不是“内存一直涨”,而是“持续增长且不回落”。单纯设阈值(如 >500MB)容易误报,比如临时大文件处理。

更稳妥的方式是监控**增长斜率 + 回落失败**:

  • 记录最近 N 次 RSS 值(例如 10 次,间隔 3 秒)
  • 用线性拟合或简单差分判断是否连续 5 次上涨,且涨幅 >5MB/次
  • 同时检查 GC 是否有效:调用 gc.collect() 后 RSS 下降
  • 报警前再 sleep 1 秒重采一次,排除瞬时毛刺

示例逻辑片段:

import gc
# ... rss_history 维护一个 deque(maxlen=10)
if len(rss_history) == 10 and all(rss_history[i] < rss_history[i+1] for i in range(5, 9)):
    gc.collect()
    time.sleep(1)
    rss_after_gc = p.memory_info().rss / 1024 / 1024
    if rss_after_gc > rss_history[-1] * 0.99:  # 未回落超 1%
        trigger_alert(rss_after_gc)

报警该怎么做才不被忽略

发个 print() 或写日志基本等于没报。真要起作用,得打通运维链路:

  • 写入本地告警文件(如 /tmp/memory_alert.pid),配合 systemd 的 PathExists 监控自动拉起处理脚本
  • 调用企业微信/钉钉 Webhook(注意加签

    名和限频,避免刷屏)
  • 向 Prometheus Pushgateway 推送指标(memory_leak_detected{pid="12345"} 1),再由 Alertmanager 发通知
  • 切忌用 smtplib 自建邮件——SMTP 很容易被当垃圾邮件拦截,且无送达反馈

为什么多线程/多进程下容易误判

psutil.Process(os.getpid()) 在子线程里仍返回主进程,没问题;但如果你用 multiprocessing,每个子进程需独立监控——父进程无法直接看到子进程 RSS。

  • 子进程启动时,手动传入监控开关和告警通道(如 queue 或 pipe)
  • 避免用 fork 后的进程继承父进程监控逻辑,因为 fork 时 RSS 会虚高(写时复制未触发)
  • 若用 concurrent.futures.ProcessPoolExecutor,建议改用 ThreadPoolExecutor 配合对象池复用,从源头减少进程级内存震荡

最常被忽略的一点:C 扩展(如 numpy、pandas、cv2)分配的内存不经过 Python GC,RSS 上涨但 gc.get_count() 完全不动——这类泄漏只能靠 RSS 趋势发现,无法靠对象计数定位。


# linux  # python  # windows  # 操作系统  # 微信  # 企业微信  # mac  # 虚拟内存  # macos  # win  # 钉钉  # 内存占用 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Python进程池调度策略_任务分发说明【指导】  Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层  Laravel怎么自定义错误页面_Laravel修改404和500页面模板  公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?  Laravel怎么集成Log日志记录_Laravel单文件与每日日志配置及自定义通道【详解】  制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?  通义万相免费版怎么用_通义万相免费版使用方法详细指南【教程】  如何在阿里云高效完成企业建站全流程?  如何在阿里云通过域名搭建网站?  Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】  jimdo怎样用html5做选项卡_jimdo选项卡html5实现与切换效果【指南】  Linux后台任务运行方法_nohup与&使用技巧【技巧】  黑客如何通过漏洞一步步攻陷网站服务器?  Laravel如何设置定时任务(Cron Job)_Laravel调度器与任务计划配置  如何用AI帮你把自己的生活经历写成一个有趣的故事?  如何在阿里云ECS服务器部署织梦CMS网站?  Laravel如何使用集合(Collections)进行数据处理_Laravel Collection常用方法与技巧  如何快速生成凡客建站的专业级图册?  微信小程序制作网站有哪些,微信小程序需要做网站吗?  如何用已有域名快速搭建网站?  网站制作企业,网站的banner和导航栏是指什么?  详解Nginx + Tomcat 反向代理 负载均衡 集群 部署指南  如何快速上传自定义模板至建站之星?  Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】  Laravel怎么连接多个数据库_Laravel多数据库连接配置  Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】  悟空识字怎么关闭自动续费_悟空识字取消会员自动扣费步骤  Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程  Laravel模型事件有哪些_Laravel Model Event生命周期详解  米侠浏览器网页背景异常怎么办 米侠显示修复  Laravel如何部署到服务器_线上部署Laravel项目的完整流程与步骤  如何快速上传建站程序避免常见错误?  Laravel如何实现API版本控制_Laravel版本化API设计方案  深圳防火门网站制作公司,深圳中天明防火门怎么编码?  Laravel用户认证怎么做_Laravel Breeze脚手架快速实现登录注册功能  如何用好域名打造高点击率的自主建站?  php在windows下怎么调试_phpwindows环境调试操作说明【操作】  大学网站设计制作软件有哪些,如何将网站制作成自己app?  Swift开发中switch语句值绑定模式  edge浏览器无法安装扩展 edge浏览器插件安装失败【解决方法】  如何用腾讯建站主机快速创建免费网站?  实例解析Array和String方法  Linux系统命令中screen命令详解  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  如何在阿里云购买域名并搭建网站?  敲碗10年!Mac系列传将迎来「触控与联网」双革新  QQ浏览器网页版登录入口 个人中心在线进入  详解MySQL数据库的安装与密码配置  php485函数参数是什么意思_php485各参数详细说明【介绍】  EditPlus中的正则表达式实战(5)