JDK中提供了命令行下的断点调试工具:JDB。它可以调试正在运行或未运行的Java程序。
现在我们编写一个Java程序
public class Hello {
private String s0 = "private";
public String s1 = "public";
public static String s2 = "static";
public static final String s3 = "static final";
public static void main(String[] args) {
String s4 = "local";
String s5 = new String("local");
s5.intern();
System.out.println(s4 == s5);
}
}
编译生成Hello.class文件。
然后命令行运行jdb:
jdb Hello
可能会出现类似下面的错误。
错误信息中,address=lslxy1021:38885引起了我们的注意。
查看hosts文件:
将lslxy1021所在行注释,问题解决。
JDB 调试
(不清楚命令的可键入 help 查看帮助信息)
首先在main方法上设置一个断点:
- stop in Hello.main
- 运行程序至断点处:run
- 运行下一行:next
这里我们先next到这里停止。
HSDB查看内存布局
新开一个窗口,首先获取刚才程序的pid。
运行jcmd -l,得到pid为269837。
接下来通过HSDB查看内存布局(针对JDK 9+版本,JDK 8中启动方式略有不同)
命令行输入命令:jhsdb clhsdb(clhsdb即hsdb的命令行版)
输入universe命令查看GC堆的地址范围和使用情况
输入scanoops start end type 查询指定类型(包括派生类),我这里输的命令是scanoops 0x00000000e3400000 0x00000000ee150000 java.lang.String,结果扫描出来的有点多。。。
我们可以用findpc address命令查看某个地址分配在哪个区域中,用inspect命令查看具体某个地址中的内容。