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