1、对象的创建:(普通对象,不包过数组和Class对象)
- new字节码指令:当Jvm遇到字节码new指令时,首先检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有必须执行类的加载过程。(双亲委派机制)
- 计算创建对象所需要的内存:类加载检查通过后,Jvm将为新生的对象分配内存,对象所需内存的大小在类加载完成后便完全确定。在堆内存中就是将一块确定大小的空闲的内存分配出来。
2、对象的内存分配:
- 如果内存时整齐连续的:如果垃圾回收机制是带有空间压缩整理的方式,那么堆内存空间就是整齐的,在给对象分配内存时就是将内存指针向空闲内存的方向偏移一下,这种分配内存的方式称之为 “指针碰撞分配对象内存”。
- 如果堆内存不是整齐的:Jvm就需要维护一张内存空闲列表,在分配内存时找一块能够满足当前对象需要的内存分配给该对象,同时记录在空间列表中。这种分配内存的方式称之为 “空闲列表分配对象内存”。
3、分配内存并发情况下保证线程安全问题:
- 问题:无论是采用指针碰撞还是空闲列表的方式分配内存,都可能存在线程安全问题,因为分配的动作和内存记录更新的动作并不是原子性的操作。
- 解决:方案一分配内存空间和更新内存记录的动作进行同步处理。Jvm是通过CAS+失败重试的方式保证分配和更新操作的原子性。
- 解决:方案二将内存分配的动作按照线程划分在不同的空间之中进行。即在堆内存中为每一个线程分配一块小空间,称之为线程分配缓冲TLAB,Jvm是否使用TLAB可以通过参数配置。
4、对象的初始化:
- 在对象头中记录一些对象的基本信息。
- 调用Class文件中的init方法,对对象进行初始化。
5、对象的内存布局:对象头 Mark Word
- 对象头是一个动态定义的数据结构,大小可变,32位或者64位。Hotspot对象头中包括两类信息:
- a.存储对象自身运行时数据,如:哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等。锁的标识位是两个bite,0和1的组合共五种。
- b.类型指针:对象指向它的类型元数据的指针,通过这个指针来确定该对象是那个类的实例。如果对象是数组,需要在对象头中记录该对象的数组长度。Jvm可以通过普通Java对象的元数据信息确定Java对象的大小。
6、对象的内存布局:实例数据
- 存储对象实例中的所有数据:程序中定义的各种类型的字段内容,无论是父类继承还是自身数据都需要记录下来。
- 字段的分配策略:相同宽度的数据类型会被分配存储在一起。
7、对象的内存布局:对齐填充
- 不是必然存在:对齐填充的作用是占位符的作用。HotSpot虚拟机的自动内存管理系统要求对象的大小必须是8的倍数,对象头的设计已经是8的倍数,所以如果实例数据不满足8的倍数,就需要对齐填充来完成补全