我们都知道,创建对象是由 new关键字调用构造方法 返回类实例(实际上还可以通过反射来创建实例)。
例如 : Person jack = new Person();
这句话到底做了什么事情呢 ? 其实就是讲对象的初始化过程。
- 1、 new 用到了Person.class,所以会先找到Person.class文件,并加载到内存中(用到类中的内容类就会被加载)
- 2、执行该对象的static代码块(静态初始块)。(如果有的话,给Person.class类进行初始化)
- 3、在堆内存中开辟空间,分配内存地址
- 4、在堆内存中建立对象特有属性,并进行默认初始化
- 5、对属性进行显示初始化(声明成员属性并赋值)
- 6、执行构造块
- 7、执行构造函数
- 8、将内存地址赋值给栈内存中的jack变量
如下图:
下面我们来看两个案例
案例一:
以下代码的输出结果是什么?
public class B {public static B t1 = new B();public static B t2 = new B();{System.out.println("构造块");}static {System.out.println("静态块");}public static void main(String[] args) {B t = new B();}
}
答案是:
构造块
构造块
静态块
构造块
答案分析:JVM先加载B这个类,初始化静态域(静态变量、静态块和静态方法,按照代码先后顺序初始化),先初始化t1,t2,所以先输出构造块 构造块;然后执行静态块,所以输出静态块;最后执行new B()输出构造块
案例二:
一下代码的输出结果是什么?
public class B {static {i = 100;}public static int i = 1;public static void main(String[] args) {System.out.println(i);}
}
答案是:1
答案分析:静态变量是在类初始化时首先被加载的,JVM会去查找类中所有的静态声明,然后分配空间,注意这时候只是完成了地址空间的分配,还没有赋值,之后JVM会根据类中静态赋值(包括静态类赋值和静态块赋值)的先后顺序来执行。对于程序来说,就是先声明了int类型的地址空间,并把地址传递给了i,然后按照类中的先后顺序执行赋值动作,首先执行静态块中i=100,接着执行i=1,那最后的结果就是i=1了。