正文
身为后端开发工程师,时刻要知道Java对象是怎么被创建的,怎么分配的以及怎么被执行引擎进行执行的。今天我们就一起来浅谈一下JVM中的Java对象。在JVM中,一个java对象并不仅仅是存储大家new出来的对象那么简单,还需要其他的信息,例如:一个java对象概括起来说分为对象头、对象体以及补齐数据部分。
其中,对象头的部分又分为Mark Word、Class Wordh和数组长度部分。数组长度部分仅仅在对象是数组类型的情况下才会存在。在一个64bit的机器环境中,一个字为8个字节,并且分配的java对象也要是8个字节的整数倍,所以产生了补齐数据,保证一个java对象占用的空间是8个字节的整数倍。Mark Word:主要用于标记对象线程的状态,以及年龄、锁信息等等。
Class Word:用于存储指向类常量池中的Class信息的指针,可以通过这个信息明确自己是哪一个对象的实例,垃圾回收器回收类信息也会检测是否存在对象指向这个类信息。
数组长度:这个部分是可有可无的,只有当一个对象是数组类型的时候,才会有这个部分记录着数组对象的长度。
对象体:也就是我们写代码创建的类以及其中的属性信息,占用空间的大小取决于对象的属性数量以及他们的类型,不同的类型占用的内存大小也是不一样的。补齐数据:为了保证一个java对象是“一个字”的整数倍。这里的“字”不同的机器架构是不一样的。如果是32bit机器,一个字就是4个字节,但是如果是64bit机器,那么一个字就是8个字节。首先一起看一下mark word区,最重要的信息如下:
这个区域记录了某一个对象的信息,例如:hash_code、age、偏向锁、锁信息。其次是Class Word区,这一个部分存储的是一个对象的类型指针,指向他的类元数据,JVM可以通过这个指针确定这个对象属于哪一个类的实例。最后对于数组长度部分,如果一个对象是数组,那么对象头还要额外的空间来存储该对象数组的长度,这个长度也是跟随着JVM的结构不同而长度不同。在32位的机器上是32bit,64位的机器上就是64bit。
当然,实际上,指针在JVM中存储时候,占用的大小是一个字的长度,不同的机器上,也是不一样的。不过,有的时候,为了节约指针带来的内存的占用,可以采用指针压缩的方式节约指针占用的空间。例如,在启动java服务的时候,可以加上+UseCompressedOops来开启指针压缩,可以将占用64bit的指针压缩到32bit。