小小面试一下
前言蜜语
最近马师傅火的不要不要的,虽然没有抢到耗子尾汁的商标注册权,但是必须得蹭一波马师傅的热度,下面就是闪电五连鞭的教学环节,你准备好了吗!
在正式内容开始前先甩两篇关于类加载机制和内存布局的文章,因为今天的内容多少与这两篇文章有直接的联系,对这方面还比较薄弱的朋友可以先看看,关注公众号Java技术栈搜索阅读JVM,在公众号回复:JVM46,还可以获取一份 46 页的 JVM 高清教程。
今天本文的内容就针对刚刚模拟面试两个问题
1.对象的创建过程
2.对象的内存布局
对象的创建过程
以下内容基于HotSpot VM 分代模型
这张图其实就能完整的说明一个对象的创建过程到底发生了什么,很多朋友可能一下看不懂,那么我们就跟着左上角的一步一步来:
-
一个对象new出来先判断线程栈是否能分配下
如果能分配下,直接分配在栈中。
如果分配不下则进行第二步。
-
判断该对象是否足够大
如果足够大,则直接进入老年代。
如果不够大,则进行第三步。
-
判断创建对象的线程的TLAB(本地线程缓冲区)空间是否足够
如果足够,直接分配在TLAB中。
如果不够,则进入Eden区中其他空间。然后进行第四步。
-
GC清除
如果清除掉了该对象,则直接结束。
如果没有清除掉对象,进行第5步。
-
此刻对象进入Survivor 1 区,判断年龄是否足够大
如果年龄足够大,则直接进入old区域。
如果年龄不够大,则进入Survivor 2 区,然后进入第4步,循环往复。
通过这张流程图和步骤解析大家应该对一个对象的创建过程有一个很清晰的概念了,但是其中还是有很多小细节会被忽略,为什么jvm会在对象的创建过程中大作文章,会分这么多种情况?为了让大家更深入的能够理解它,我们就再来看看下面这几个问题:
- 为什么对象会选择先分配在栈中?
首先栈是线程私有的,将对象优先分配在栈中,可以通过pop直接将对象的所有信息,空间直接清除,当线程消亡的时候也可以直接清理这一块儿TLAB区域。
- 为什么jvm会让大对象会直接进入老年代?
大对象需要连续的空间来存储,如果不存入老年代对jvm说就可能是一个负担,如果没有足够的空间就有可能导致提前触发gc来清理空间来安置大对象。
- 为什么会选择先进入TLAB?
TLAB是线程本地缓冲区,TLAB的好处就是防止不同线程创建对象选择同一块儿内存区域而产生竞争,会使其概率大大减少。
- 为什么会有两个Survivor区?并且存活且年龄不够大的对象会从一个Survivor区转到另一个Survivor区?
根据根可达算法,jvm会从开始寻找到所有正在使用的对象,没有使用的就是垃圾,通常情况下,很多对象都是用完就抛弃的,所以真正在Survivor区长时间存活的对象非常少,将这部分对象从一个Survivor区转到另一个Survivor区后,就可以直接对这个Survivor区进行全量的空间回收了,效率会很高。
对象的内存布局
作者可不是标题党,哈哈,所以我们回到文章的标题,Object o = new Object();到底占用多少个字节?
这道题的目的其实就是考验看你对对象的内存布局了解的是否清晰,先上图:
在java中对象的内存布局分为两种情况,非数组对象和数组对象,数组对象和非数组对象的区别就是需要额外的空间存储数组的长度length。
对象头
对象头又分为MarkWord和Class Pointer两部分。
MarkWord:包含一系列的标记位,比如轻量级锁的标记位,偏向锁标记位,gc记录信息等等,在32位系统占4字节,在64位系统中占8字节。
ClassPointer:用来指向对象对应的Class对象(其对应的元数据对象)的内存地址。在32位系统占4字节,在64位系统中占8字节。
Length:只在数组对象中存在,用来记录数组的长度,占用4字节
Interface data
- Interface data:对象实际数据,对象实际数据包括了对象的所有成员变量,其大小由各个成员变量的大小决定。(这里不包括静态成员变量,因为其是在方法区维护的)
Padding
- Padding:Java对象占用空间是8字节对齐的,即所有Java对象占用bytes数必须是8的倍数,是因为当我们从磁盘中取一个数据时,不会说我想取一个字节就是一个字节,都是按照一块儿一块儿来取的,这一块大小是8个字节,所以为了完整,padding的作用就是补充字节,保证对象是8字节的整数倍。
特意标注了32位系统和64位系统不同区域占用空间大小的区别,这是因为对象指针在64位JVM下的寻址更长,所以想比32位会多出来更多占用空间。
但是现在假设一个场景,公司现在项目部署的机器是32位的,你们老板要让你将项目迁移到64位的系统上,但是又因为64位系统比32位系统要多出更多占用空间,怎么办,因为正常来说我们是不需要这一部分多余空间的,所以jvm已经帮你考虑好了,那就是指针压缩。
指针压缩
-XX:+UseCompressedOops 这个参数就是JVM提供给你的解决方案,可以压缩指针,将占用的空间压缩为原来的一半,起到节约空间的作用,classpointer参数大小就受到其影响。
Object o = new Object()到底占用多少个字节?
通过刚才内存布局的学习后,这个问题就很好回答了,面试官其实就是想问你对象的内存布局是怎样的,我们这里就针对这个问题的结果分析下,这里分两种情况:
在开启指针压缩的情况下,markword占用8字节,classpoint占用4字节,Interface data无数据,总共是12字节,由于对象需要为8的整数倍,Padding会补充4个字节,总共占用16字节的存储空间。
在没有指针的情况下,markword占用8字节,classpoint占用8字节,Interface data无数据,总共是16字节。
结语
今天的文章和大家介绍了一个对象的创建过程,从它的出生到死亡,都经历了什么?
也和大家详细的说明了对象的内存布局,深入解剖了一下对象的身体构造,这下面试官再问你,可就有的聊了。