top/htop 显示 VIRT 非常大但 RES 不高是否真的是内存泄漏

12次阅读

VIRT大而RES不高通常不是内存泄漏,而是虚拟地址空间分配未实际使用;VIRT含未映射内存,RES才是真实物理内存占用;需警惕VIRT与RES同步持续增长、匿名映射不释放等异常。

top/htop 显示 VIRT 非常大但 RES 不高是否真的是内存泄漏

不是,VIRT 很大但 RES 不高,通常不是内存泄漏,而是进程申请了大量虚拟地址空间,但尚未真正使用或映射到物理内存。

理解 VIRT 和 RES 的本质区别

VIRT 是进程“可访问”的全部虚拟地址空间大小,包括:已分配但未写入的内存、共享库、mmap 映射区域、预留但未使用的地址段等。它不消耗实际物理内存。

RES 是当前真实占用的物理内存页(驻留集),包含已实际写入的堆/、代码段、部分共享库页等。它是系统真正紧张的资源。

例如:malloc(1GB) 后立即 free,VIRT 可能短暂飙升又回落;但如果只 malloc(1GB) 却从不写入任何字节,VIRT 会增加,RES 几乎不变——这完全正常,不算泄漏。

什么情况下 VIRT 持续增长才需警惕

  • VIRT 长期单向增长(如数小时/天),且伴随 RES 同步缓慢上升 → 可能存在堆分配未释放或 mmap 泄漏
  • VIRT 增长后不再收缩,而进程反复创建大量对象(尤其在 C/c++ 中调用 mmap/MAP_ANONYMOUS 未 munmap)→ 检查匿名映射段
  • java/go 等带 GC 的语言中,VIRT 大常因堆外内存(Netty Direct Buffer、JNI 分配)未释放,此时 RES 也可能滞后上涨

快速验证是否真泄漏的三步操作

  • pmap -x 查看 anon-rss 和 mapped 文件页占比,重点关注 anon(匿名映射)列是否随时间增长
  • 运行 cat /proc//smaps | awk '/^Size:/ {size+=$2} /^RSS:/ {rss+=$2} END {print "Size:", size/1024, "MB; RSS:", rss/1024, "MB"}',观察 RSS 是否稳定
  • 对比多次 htop 快照中 RES 曲线:若 RES 在业务低峰期仍不回落(比如 30 分钟内下降

容易被误判的常见场景

  • JVM 设置了超大堆(-Xmx32g),即使只用了 2GB,VIRT 也会接近 32g + 元空间 + 直接内存 → 正常现象
  • glibc 的 malloc 使用 mmap 分配大块内存(默认 ≥128KB),这些区域计入 VIRT,但未 touch 前不占 RES
  • 程序加载大量动态库或使用 hugepage 预分配,VIRT 高但 RES 合理
text=ZqhQzanResources