一、引言
如你所知,Java是一门面向对象的编程语言。我们平常在写代码的时候也是在不停的操作各种对象,那么当你在写出User user = new User();这样一行代码的时候,JVM都做了些什么呢?
二、了解对象
1、内存布局
在Hotspot虚拟机中一个对象的内存布局分为三个部分:对象头、实例数据、对齐填充。
对象头又有两部分的信息,第一部分是用于存储对象自身的运行数据(HashCode、GC分代年龄、锁状态标志等)。另一部分是类型指针,指向它的类元数据,虚拟机通过这个指针确定这个对象是哪个类的实例(如果使用句柄池方式则不会有)。如果是数组还会有一个记录数组长度的如下表所示:Mark Word是一个非固定的数据结构以便在极小的空间内存储尽量多的信息,它会根据对象的状态复用自己的存储空间。各状态下的存储内容如下表所示:
2、对象的访问
Java程序中我们操作一个对象是通过指向这个对象的引用。我们都知道对象存在堆中,这个引用存在虚拟机栈中。那么引用通过什么方式去定位堆中对象的位置呢?
三、创建对象流程
上面介绍了对象的基本信息,现在来讲一讲创建对象的流程:
分配内存有两种方式:
Java堆内存是规整的(使用标记整理或带压缩的垃圾收集器),使用一个指针指向空闲位置,分配内存既将指针移动与分配大小相等的距离内存不是规整的(使用标记清除的垃圾收集器),虚拟机维护一个可用内存块列表,分配内存时从列表中找到一个足够大的内存空间划分给对象并更新可用内存列表。无法找到足够的内存时会触发一次GC
分配内存时并发问题解决方案:
对分配内存空间的动作进行同步操作---采用CAS失败重试的方式保证更新操作的原子性。每个线程在堆中预先分配一块小内存,称为本地线程分配缓冲(Thread Local Allocation Buffer,TLAB),哪个线程要分配内存就在它的TLAB上分配,只有TLAB用完并分配新的TLAB时才需要同步锁定。通过-XX:+/-UseTLAB参数来设定。四、创建对象指令重排序问题
A a = new A();
new一个对象的简单分解动作:
分配对象的内存空间初始化对象设置引用指向分配的内存地址其中2、3两步间会发生指令重排序,导致多线程时如果在初始化之前访问对象则会出现问题,单例模式的双重检测锁模式正是会存在这个问题。可以使用volatile来禁止指令重排序解决问题
链接:https://juejin.im/post/5da971216fb9a04e17209328
关注我了解更多程序员资讯技术,领取丰富架构资料