1、类加载:在首次创建该类的实例之前,需要先加载该类的字节码文件。这一步会涉及到类加载器的工作,将类文件加载到内存中,并创建对应的 Class 对象。
2、分配内存:在 Java 堆(Heap)中分配内存空间用于存储对象的实例数据。Java 堆是在 JVM 启动时创建的,用于存放所有的对象实例。
存在的问题:在并发情况下,可能发生多个线程同时分配内存的情况。
- 对象指针竞争:多个线程同时使用原来的指针来分配内存。这可能导致多个线程在同一时间分配到相同的内存空间,从而产生数据混乱或错误。
- 内存地址还未改变:在一个线程正在为对象 A 分配内存时,另一个线程可能同时使用相同的指针来分配内存给对象 B,此时对象 A 的内存地址还未改变。这种情况可能导致对象 B 覆盖或修改对象 A 的数据,引发数据一致性问题。
解决方案:CAS或者TLAB(Thread Local Allocation Buffer)
- 本地线程分配缓冲(TLAB): 每个线程分配了一块小的缓冲区空间,让线程将新创建的对象存储在自己的 TLAB 中,减少了线程间的竞争,提高了对象分配的性能。当线程的 TLAB 空间不足以分配对象时,会使用 CAS 机制来进行分配,避免了多线程分配对象时的竞争问题。(通过-XX:+UserTLAB开启 -XX:-useTLAB关闭 JVM默认开启 可以通过-XX:TLABSize设置TLAB大小 )
3、设置对象头:对象头(Object Header):包括对象的标记字段、类型指针、数组长度等信息。
- 标记字段(Mark Word):标记字段用于垃圾回收器标记和追踪对象的状态和生命周期。它可以包含一些标志位,如对象的存活状态、锁定状态、同步信息等。
- 类型指针(Klass Pointer):类型指针指向对象所属的类的元数据信息。通过类型指针,JVM可以确定对象的所属类,从而进行方法的调用和字段的访问等操作。
- 数组长度(Array Length):如果对象是一个数组,那么对象头还会包含数组的长度信息。长度信息可以帮助JVM遍历数组的元素,进行边界检查等操作。
4、指针压缩:指针压缩是一种优化技术,在32位虚拟机中尤为重要。在32位系统下,对象引用占用4个字节,为了节省内存空间,在指针压缩技术下,虚拟机会对对象引用进行压缩,将原本指向对象堆的绝对地址转换为相对于某个基准地址的偏移量。这样可以有效减少内存占用,提升内存使用效率。
5、执行init()方法:init方法包含成员变量、构造代码块的初始化,按照声明的顺序执行。
6、执行构造方法:执行对象的构造方法。至此,对象创建成功。
ps:以下是我整理的java面试资料,感兴趣的可以看看。最后,创作不易,觉得写得不错的可以点点关注!