public class JVM1 {
public static void main(String[] args) {
Singleton s = Singleton.getInstantce();
System.out.println(“counter1:” +s.counter1);
System.out.println(“counter2:” +s.counter2);
}
}
class Singleton {
private static Singleton testCase2 = new Singleton();
public static int counter1;
public static int counter2 = 0;
private int counter3 = 4;
private Singleton() {
System.out.println(“counter3:” +counter3);
counter1++;
counter2++;
}
public static Singleton getInstantce() {
return testCase2;
}
}
代码如上
看到代码后我们大多数人不假思索的写出输出结果:
counter3:4
counter1:1
counter2:1
但是结果真的是这样么?
如果你有疑惑可以试试,正确结果应为
counter3:4
counter1:1
counter2:0
我们来简单的分析一下:
JVM类加载机制:加载,连接,初始化。
1、由于Singleton.getInstantce()会主动初始化Singleton类,连接的准备阶段counter1、counter2 会分配内存并设置默认值0,而testCase2则设置为null。
2、类的初始化阶段需要做是执行类构造器(类构造器是编译器收集所有静态语句块和类变量的赋值语句按语句在源码中的顺序合并生成类构造器,对象的构造方法是(),类的构造方法是(),可以在堆栈信息中看到),因此先执行第一条静态变量的赋值语句即Singleton testCase2= new Singleton(),此时会进行对象的初始化,对象的初始化是先初始化成员变量再执行构造方法,所以会先进行counter3 = 4;在调用Singleton构造方法,接着执行类的初始化
为了验证我们的猜想,我们不妨调整一下类变量的顺序,如下
public class JVM1 {
public static void main(String[] args) {
Singleton s = Singleton.getInstantce();
System.out.println("counter1:"+s.counter1);
System.out.println("counter2:"+s.counter2);
}
}
class Singleton {
public static int counter1;
public static int counter2 = 0;
private static Singleton testCase2 = new Singleton();
private int counter3 = 4;
private Singleton() {
System.out.println(“counter3:” +counter3);
counter1++;
counter2++;
}
public static Singleton getInstantce() {
return testCase2;
}
}
可以试试输出结果为:
counter3:4
counter1:1
counter2:1
类的初始化阶段需要做是执行类构造器(类构造器是编译器收集所有静态语句块和类变量的赋值语句按语句在源码中的顺序合并生成类构造器,对象的构造方法是(),类的构造方法是(),可以在堆栈信息中看到),因此先执行静态变量的赋值语句即public static int counter1;public static int counter2 = 0;
先执行类的初始化,此时counter1和counter2变成了0;接着执行对象的构造Singleton.getInstantce()
构造此时会进行对象的初始化,对象的初始化是先初始化成员变量再执行构造方法,所以会先进行counter3 = 4;