首先代码已经被我用我自己写的一个小demo模拟展示了出来(用我们生产环境的代码太复杂,大家也看不懂)
public class Singleton {
private static Singleton singleton = new Singleton();
private static int i = 7;
private static Date currentDate = new Date();
public static Singleton getInstance(){
return singleton;
}
private Singleton(){
printMsg();
}
public static void printMsg(){
System.out.println(i);
System.out.println(currentDate);
}
}
public class Test {
public static void main(String[] args) {
Singleton s = Singleton.getInstance();
s.printMsg();
}
}
大家分析下最后输出的是什么?
===========================================不给看==================================================
===========================================不给看==================================================
===========================================不给看==================================================
===========================================不给看==================================================
===========================================不给看==================================================
答案:
0
null
7
Fri Feb 21 22:10:33 ICT 2014(此结果因时间而异)
2.原因分析
想到类的初始化,免不了那些类初始化顺序的问题,首先我在这纠正我自己的一个错误,在我的印象中类的初始化顺序是这样的:
static字段>static代码块>实例变量>非静态代码块(这样是错误的)
正确的顺序应该是这样的:
(static字段,static代码块)>(实例变量,非静态代码块),括号内的两者初始化的先后顺序依赖于两者在代码中的顺序,不一定谁在前谁在后。
接下来再分析下以上这段代码:
1.程序执行 Singleton s = Singleton.getInstance();首先要初始化Singleton,在初始化类Singleton时,第一句首先是new了自身,此时会调用构造函数,构造函数中打印了另外两个静态变量的值,但是它们两个静态变量并没有初始化,还是各自的默认值,分别为0和null,所以打印的是这两个结果。接下来的执行情况相信大家都很明白了~~~
我想说的主要是这一点,一个类在new了自身的一个静态变量,并且构造函数中有对其他静态变量的使用,这时候要特别注意静态变量的顺序。具体的初始化过程大体看了下深入JVM那本书,过程是相当复杂的,这个等我研究一阵子自后再写另一篇博客吧。