java对象创建步骤和对象内存布局

一、对象的创建

1.查看指令是否加载
当虚拟机遇到一条new指令时,首先先去检查这个指令的参数能否能在元空间常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、机械和初始化过(即判断类元信息是否存在)。如果没有,那么就在双亲委派模式下,通过类加载器加载class文件,并生成对应class类对象

2.分配内存
在类加载完成后就可以完全确定对象所需内存了,这时内存分配可以分为两种,java堆内存规整和不规整。java堆是否完整取决于垃圾收集器是否带有压缩整理功能。
⑴内存规整----->指针碰撞:用过的内存放一边,没用过的内存放一边,中间放一个指针作为分几点的指示器。分配内存时就把指针向空闲控件挪一段距离就好。
⑵不规整------>虚拟机会维护一个列表,记录那些内存可用。在分配内存的时候会找一块足够大的内存空间划分给对象实例,然后更新列表。

线程安全
在对象创建的时候,分配内存时可能出现并发问题。正在给对象A分配内存,指针还没修改,对象B又同事使用原来指针分配内存。
解决方案有两种:
1.对分配空间的动作进行同步处理
2.把内存分配动作按照线程划分在不同的空间中进行。假如有两个线程都在创建对象,那就让每个线程在java堆中预先分配一块内存,称为本地线程分配缓冲(TLAB),哪个线程要分配内存,就在哪个线程的TLAB上分配,只有TLAB用完并分配新的TLAB时才需要同步。

3.默认初始化
所有属性赋默认值

4.设置对象头
设置MarkWord和Klass Point(类型指针)的值

5.执行进行初始化
对对象的属性进行显示初始化,代码块初始化,构造函数初始化

二 、对象的内存布局

对象中存储布局分为对象头、梳理数据、对齐填充。

1.对象头
对象头有两部分:
第一部分存储对象自身运行时数据如哈希码,GC分代年龄、锁状态标志等等。在32位和64位虚拟机中为32位或者64位。其实这些空间不够用,只是被设计成一个非固定的数据结构以便在极小的空间内存储尽量多的信息,会根据对象装固态复用自己的存储空间。例如在未被锁的状态下和锁状态下,存储各个信息的位数不一样。
第二部分是类型指针,指向雷院数据的指针,通过这个指针确定是哪个类的实例。如果对象是一个java数组,还必须要有一块记录数组长度的数据,因为不能通过数组类元数据中确定数组大小。

指针访问.png

2.实例数据
是对象真正的村塾的有效信息,也是代码中定义的字段内容,无论是从父类继承还是子类定义,都要记录。Hotspot默认分配策略为long/double、int、short/char、byte/boolean
相同宽度的字段会被分配到一起。在满足这个条件以前,父类定义的变量会出现在子类之前。

3.对齐填充
占位符的作用,要求对象起始地址必须是8字节的整数倍,也就是对象大小必须是8字节整数倍。对象头部分一般都是满的,但是对象实例数据可能会有没对齐的,就需要来补全

你可能感兴趣的:(java对象创建步骤和对象内存布局)