VIRT大而RES不高通常不是内存泄漏,而是虚拟地址空间分配未实际使用;VIRT含未映射内存,RES才是真实物理内存占用;需警惕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/,观察 RSS 是否稳定/smaps | awk '/^Size:/ {size+=$2} /^RSS:/ {rss+=$2} END {print "Size:", size/1024, "MB; RSS:", rss/1024, "MB"}' - 对比多次
htop快照中 RES 曲线:若 RES 在业务低峰期仍不回落(比如 30 分钟内下降
容易被误判的常见场景
- JVM 设置了超大堆(-Xmx32g),即使只用了 2GB,VIRT 也会接近 32g + 元空间 + 直接内存 → 正常现象
- glibc 的 malloc 使用 mmap 分配大块内存(默认 ≥128KB),这些区域计入 VIRT,但未 touch 前不占 RES
- 程序加载大量动态库或使用 hugepage 预分配,VIRT 高但 RES 合理










