专注虚拟机与编译器研究

第5.5篇-使用HSDB查看类变量的内存布局

如下的实例来自Hotspot 垃圾回收之oop_iterate(一) 源码解析,我做了一些微小的改动。

查看如下实例的内存布局:

package com.test;

public class Parent {
	private   Integer     a = 1;
	protected long        b = 2;
	protected final short c = 3;
	public    char        d ='A';
}  

Son类继承Parent类,如下:

package com.test;

public class Son extends Parent {
	private   Integer     a = 11;
	protected long        b = 12;
	protected final short c = 13;
	public    char        d ='B';
	
	public static void main(String args[]){
		Son s = new Son();
		System.out.println(s.a);
	}
}

连接上HSDB后,在控制台输入universe命令,查看当前Java进程的堆内存,如下所示:

hsdb> universe
Heap Parameters:
Gen 0:
  eden [0x00000000ff600000,0x00000000ff680030,0x00000000ff8b0000) space capacity = 2818048, 18.606354469476745 used
  from [0x00000000ff8b0000,0x00000000ff8b0000,0x00000000ff900000) space capacity = 327680, 0.0 used
  to [0x00000000ff900000,0x00000000ff900000,0x00000000ff950000) space capacity = 327680, 0.0 usedInvocations: 0
Gen 1: old [0x00000000ff950000,0x00000000ff950000,0x0000000100000000) space capacity = 7012352, 0.0 usedInvocations: 0

接着输入如下命令搜索对象的起始地址:

scanoops 0x00000000ff600000 0x0000000100000000 com.test.Son

命令scanoops查看指定类型的实例对象,接受两个必选参数和一个可选参数:必选参数是要扫描的地址范围,一个是起始地址一个是结束地址;可选参数用于指定要扫描什么类型的实例对象。

得到表示Son实例的instanceOop的内存首地址,如下:

0x00000000ff670190 com/test/Son

使用inspect命令查看这个地址处的oop的全部数据,如下:

hsdb> inspect 0x00000000ff670190
instance of Oop for com/test/Son @ 0x00000000ff670190 @ 0x00000000ff670190 (size = 48)
_mark: 1
_metadata._compressed_klass: InstanceKlass for com/test/Son
a: Oop for java/lang/Integer @ 0x00000000ff62e970 Oop for java/lang/Integer @ 0x00000000ff62e970
b: 2
c: 3
d: 'A'
a: Oop for java/lang/Integer @ 0x00000000ff62ea10 Oop for java/lang/Integer @ 0x00000000ff62ea10
b: 12
c: 13
d: 'B'

使用了指针压缩,如果不想使用指针压缩,可以使用如下jdb命令:

jdb -XX:+UseSerialGC -Xmx10m -XX:-UseCompressedOops

另外还可以使用图形化的方式查看内存布局。单击HSDB的Tools->Inspector菜单选项,输入地址即可,如下图所示。

 

单击HSDB工具栏的Tools->Class Brower命令,输入Parent和Son后,便能查看类中所定义的各个字段的偏移量,如下图所示。

 

不进行指针压缩,使用如下命令:

jdb -XX:+UseSerialGC -Xmx10m -XX:-UseCompressedOops

使用univers、scanoops与inspector命令查看对象的内存布局,如下:

hsdb> inspect 0x00007f0ae468b198
instance of Oop for com/test/Son @ 0x00007f0ae468b198 @ 0x00007f0ae468b198 (size = 64)
_mark: 1
_metadata._klass: InstanceKlass for com/test/Son
a: Oop for java/lang/Integer @ 0x00007f0ae463d4d8 Oop for java/lang/Integer @ 0x00007f0ae463d4d8
b: 2
c: 3
d: 'A'
a: Oop for java/lang/Integer @ 0x00007f0ae463d5c8 Oop for java/lang/Integer @ 0x00007f0ae463d5c8
b: 12
c: 13
d: 'B'

使用图形化的方式查看内存布局,如下图所示。

 查看类中所定义的各个字段的偏移量,如下图所示。

 

除了使用HotSpot虚拟机命令、HSDB外,还可以使用JOL等来查看字段的内存布局,使用比较简单,这里不在介绍。

参考文章:

(1)终于我用JOL打破了你对java对象的所有想象 

(2)Hotspot 垃圾回收之oop_iterate(一) 源码解析

 

 

 

 

  

posted on 2020-08-08 07:11  鸠摩(马智)  阅读(744)  评论(0编辑  收藏  举报

导航