49.JVM--对象的实例化内存布局
● 对象实例化:
● 对象创建的方式
1、new:最常见的方式、单例类中调用getInstance的静态类方法,XXXFactory的静态方法
2、Class的newInstance方法:在JDK9里面被标记为过时的方法,因为只能调用空参构造器,并且权限必须为 public
3、Constructor的newInstance(Xxxx):反射的方式,可以调用空参的,或者带参的构造器
4、使用clone():不调用任何的构造器,要求当前的类需要实现Cloneable接口中的clone方法
5、使用序列化:序列化一般用于Socket的网络传输
6、第三方库
● 创建对象的步骤:
1、判断对象对应的类是否加载、链接、初始化
1)虚拟机遇到一条new指令,首先去检查这个指令的参数能否在Metaspace的常量池中定位到一个类的符号引用,
并且检查这个符号引用 代表的类是否已经被加 载,解析和初始化。(即判断类元信息是否存在)。
2)如果该类没有加载,那么在双亲委派模式下,使用当前类加载器以ClassLoader + 包名 + 类名为key进行查找对应的.class文件,
如果没有找到文件, 则抛出 ClassNotFoundException异常,如果找到,则进行类加载,并生成对应的Class对象。
2、为对象分配内存
3、处理并发问题(堆空间绝大部分空间是共享的,多个线程访问会有线程安全的问题)
1)采用CAS+失败重试保证更新的原子性
2)每个线程预先分配TLAB
4、初始化分配到的内存
所有属性设置默认值,保证对象实例字段在不赋值可以直接使用
5、设置对象的对象头
将对象的所属类(即类的元数据信息)、对象的HashCode和对象的GC信息、锁信息等数据存储在对象的对象头中。这个过程的具体设置方式取决于JVM实现。
6、执行init方法进行初始化
在Java程序的视角看来,初始化才正式开始。实例成员赋值,执行实例化代码块,调用类的构造方法,并把堆内对象的首地址赋值给引用变量
● 对象的内存布局
● 对象头:
对象头包含两部分:运行时元数据和类型指针
1、运行时元数据:
1)哈希值(HashCode),可以看作是堆中对象的地址
2)GC分代年龄(年龄计数器)
3)锁状态标志
4)线程持有的锁
2、类型指针:
指向类元数据,确定该对象所属的类型。指向的其实是方法区中存放的类元信息
● 实例数据
它是对象真正存储的有效信息,包括程序代码中定义的各种类型的字段(包括从父类继承下来的和本身拥有的字段)
● 对齐填充
不是必须的,也没特别含义,仅仅起到占位符的作用