如下问题可能会被问:
对象在内存中是怎么布局的?
synchronized锁,锁住的是对象?锁在对象的哪里?
如何计算一个对象所占内存的大小?
对象的布局
一个对象在堆区的布局可以分为三块:对象头、实例数据、对齐填充。
对象头分为三个部分:mark word、Kclass point(类型指针,指向类信息,来表明是哪个类的实例) 、数据长度(仅仅是数组对象才会有)
mark word 中有什么?
以上是64位JVM下的mark word。有几个关键标志需要注意:分代年龄、锁的标志位、指向互斥锁(重量级锁)的指针(monitor监控器),通过这几个指标,大概知道以下几个问题:
- 超过15岁(默认)的对象会被放入老年代,原来计数的标志位是在对象头的markword里
- 可以设置锁的标志位,将锁设置为无锁-偏向锁-自旋锁-重量级锁
- 指向互斥锁(重量级锁)的指针,其实就是指向的monitor监视器
通过以上分析,锁是对象。给对象加锁,就是修改其对象头上的标志位,至于重量级锁,需要获取一个monitor监视器
如何计算对象的大小
对象内存=对象头+实例数据+对齐填充
对象头:我们以虚拟机为64位的机器为例,那么对象头占用的内存是8(markword)+4(Kclass point类型指针)=12Byte。如果是数组的话那么就是16Byte。
对齐填充:起到占位符的作用,对象的内存字节数是8的整数倍。不够的话需要填充字节。
String str="",这个占用多少字节?
首先String的构造(int hash和一个char数组)
String 对象=对象头(12)+实例化数据(4(int)+4(char[]指针))+填充(4)=24(原来是20 不满足,填充了4个字节)。
所以String对象占用24字节,那么空的String对象占用=24+char[]数组(16(对象头))=40。
String str=“abcdefg”,这个占用多少字节?
那么str的内存=40+7*2(char 占两个字节)=54 +2(填充)=56
上述的填充数据其实应该放到最后的才加上的,我这样写是为了方便理解