jhat 已被废弃多年,jdk 9 起彻底移除,依赖过时 http 服务器和 swing 类,模块化后无法加载 sun.tools.jhat.jhat;现代替代方案是 jvisualvm(jdk 8 自带,jdk 9+ 需单独下载)或 jcmd/jmap 快速分析。

jhat 已被废弃多年,别再用它分析堆转储。JDK 9 起彻底移除,JDK 7–8 中也仅作兼容保留;现在打开 jhat 只会遇到 ClassNotFoundException 或直接报错退出,不是你配置错了,是它本来就不可用了。
为什么 jhat 现在基本跑不起来
它依赖 JDK 内置的过时 HTTP 服务器和 Swing UI 类,在模块化(Jigsaw)后无法加载核心类如 sun.tools.jhat.JHat;即使强行用 JDK 8 运行,也会因默认启用 TLS 1.2+ 而与内置 HTTP 服务冲突,浏览器访问 http://localhost:7000 时常卡白屏或拒绝连接。
- JDK 9+:运行即报
Exception in thread "main" java.lang.NoClassDefFoundError: sun/tools/jhat/JHat - JDK 8u202+:可能启动成功但页面 JS 报
TypeError: Can't find variable: heap - 所有版本:不支持现代堆格式(如 G1 的
hprof压缩块、ZGC 的元数据结构)
替代方案:用 jvisualvm 或 jdk.jfr + VisualVM 打开 .hprof
真正能干活的是 jvisualvm(JDK 8 自带,JDK 9+ 需单独下载),它能实时解析堆快照、按类/实例/引用链过滤,比 jhat 的静态 HTML 快 5 倍以上,且支持 OQL 查询。
- 启动命令:
jvisualvm --openfile your-app.hprof - 关键操作:点「Heap Walker」→「Classes」→ 右键类名 → 「Show Instances」→ 查看具体对象引用路径
- 注意:JDK 17+ 默认不带
jvisualvm,需从 https://www.php.cn/link/c65db23b93631916f5bb31abf24462b8 下载独立版,选匹配 JDK 版本的 release
命令行轻量替代:用 jcmd + jmap 快速定位大对象
如果只是想确认某次 OOM 是不是由某个类暴增导致,根本不用生成全量堆报告——jmap -histo 三秒出结果,还避开了 jhat 的内存爆炸风险(它自己常吃掉 4GB+ 堆)。
立即学习“前端免费学习笔记(深入)”;
- 查存活对象数量:
jmap -histo:live <pid> | head -20</pid> - 导出简略堆快照(不落地文件):
jcmd <pid> VM.native_memory summary</pid> - 对比两次快照差异:用
diff ) ),关注instance_count和bytes列突变
真正麻烦的从来不是“怎么打开堆文件”,而是“打开后怎么看懂引用链是否合理”。jhat 生成的 HTML 里一堆 java.lang.Object@0x7f8a1c002b00,没上下文,没法反查业务代码位置;而 VisualVM 点击实例能直接跳转到创建它的栈帧——这个能力,jhat 从没实现过。









