判断是否加载、分配内存(指针碰撞或者空闲链表)、初始化为零值、设置对象头(实例是哪个类的实例、类的元信息位置、GC 分代年龄等)、init 方法。
Java 虚拟机创建一个对象包含以下步骤:
-
- 给对象分配内存;
-
- 将对象的实例变量自动初始化为其变量类型的默认值;
-
- 初始化对象,给实例变量赋予正确的初始值。
针对第三个步骤,JVM 可采用三种方式来初始化对象,采用何种方式取决于创建对象的方式:
-
- 如果对象是通过 clone() 方法创建的,那么 JVM 把原来被克隆的对象的实例变量的值拷贝到新对象中;
-
- 如果对象是通过 ObjectInputStream 类的 readObject() 方法创建的,那么 JVM 通过从输入流中读入的序列化数据来初始化那些非暂时性(non-transient)的实例变量;
-
- 如果实例变量在声明时被显式初始化,那么就把初始化值赋给实例变量,接着再执行构造方法。这是最常见的初始化对象的方式。
总结对象创建过程:
- 首次创建对象时,类中的静态方法/静态字段首次被访问时,Java 解释器必须先查找类路径,以定位 .class 文件;
- 然后载入 .class(这将创建一个 Class 对象),有关静态初始化的所有动作都会执行。因此,静态初始化只在 Class 对象首次加载的时候进行一次;
- 当用 new 方法创建对象时,首先再堆上为对象分配足够的存储空间;
- 这块存储空间会被清零,这就自动地将对象中的所有基本类型数据都设置成了缺省值(对数字来说就是 0,对 boolean 和 str 也相同),而引用则被设置成了 null;
- 执行所有出现于字段定义处的初始化动作(非静态对象的初始化);
- 执行构造器。
init 方法
Java 在编译之后会在字节码文件中生成 init 方法,称之为实例构造器,该实例构造器会将语句块,变量初始化,调用父类的构造器等操作收敛到 init 方法中,收敛顺序为:
- 父类变量初始化
- 父类语句块
- 父类构造函数
- 子类变量初始化
- 子类语句块
- 子类构造函数
- 收敛到 init 方法的意思是:将这些操作放入到 init 中去执行。
原文
https://juejin.im/post/5d99d651f265da5b6b62fe6b