这里只考虑在一个类的情况下类初始化和对象实例化的顺序,不考虑有父类的情况下。
看下面这段代码:
package com.huang.jvm;
/**
* @author hsz
*/
public class Demo {
int a = print(3);
public Demo() {
System.out.println("构造方法");
}
{
System.out.println("非静态块");
}
int b = print(4);
public int print(int value) {
System.out.println("实例" + value);
return value;
}
static int i = printStatic(1);
static {
System.out.println("静态块");
}
static int j = printStatic(2);
public static int printStatic(int value) {
System.out.println("静态" + value);
return value;
}
public static void main(String[] args) {
new Demo();
}
}
执行结果如下:
静态1
静态块
静态2
实例3
非静态块
实例4
构造方法
Java虚拟机启动时,会先对定义了main方法的类进行加载,而类加载过程会进行加载、验证、准备、解析和初始化,而初始化过后就可以使用该类了。在准备阶段中,虚拟机首先会为类变量分配内存并赋初始值,int型的变量初始值是0,之后会在初始化阶段为变量赋予真正的要设置的值,并且也会执行static块中的内容,至于哪个先执行,则看赋值操作和static块的位置,哪个在前哪个先执行。所以以上的执行结果中依次打印出静态1,静态块和静态2。
注意:类加载在类的整个生命周期中只会进行一次,除非类卸载后又重新加载。
类初始化后就可以执行main方法中的代码了。当遇到new Demo()时,会进行对象实例化。先为成员变量赋默认的初始值(类似于类加载过程中的准备阶段为静态变量赋初始值值),然后再按照成员变量以及非静态块的出现顺序进行初始化,等所有成员变量初始化完成以及非静态块执行完之后调用构造方法完成对象的创建。每次new一个对象时都会执行一遍上面实例化的过程。所以类加载完后上面代码的执行结果中先打印实例2,非静态块和实例4,最后在执行构造方法。