Java的静态变量初始化的坑

在网上看到一个很有意思的题目,题目如下

class SingleTon {  
    private static SingleTon singleTon = new SingleTon();  
    public static int count1;  
    public static int count2 = 1;  
  
    private SingleTon() {  
        count1++;  
        count2++;  
    }  
  
    public static SingleTon getInstance() {  
        return singleTon;  
    }  
}  
  
public class Test {  
    public static void main(String[] args) {  
        SingleTon singleTon = SingleTon.getInstance();  
        System.out.println("count1=" + singleTon.count1);  
        System.out.println("count2=" + singleTon.count2);  
    }  
}  

最开始的时候以为非常简单就是  count1=1 和 count2=2,但是实际在eclipse里面运行一下才发现答案是错的

这是就涉及到实例的初始化流程

1.类被加载的时候,普通方法加载到方法区,静态方法和静态字段加载到方法区中的静态区

2.首先静态字段会进行默认初始化。即 singTon=null       count1=0        count2=0

3.然后静态字段会进行显示初始化。问题就出现这里(这里就是影响结果的地方)

4.首先 singTon 进行显示初始化 它会创建 实例,调用构造函数,执行完成以后 此时 count1=1   count2=1

5.然后才是静态变量 count1和count2进行显示初始化,因为count1没有显示初始化值,所以结果就是 count1=1 但是 count2变量进行完显示初始化后值就为 1了  (在此过程之前静态变量count1和count2还没有进行显示初始化的)

6.然后执行静态代码块中的内容,此处没有静态代码块。

所以,综上 结果是 singTon.count1=1  singTon.count2=1

 

如果是以下这种情况

class SingleTon {  
    public static int count1;  
    public static int count2 = 1; 
    private static SingleTon singleTon = new SingleTon();  
    
    private SingleTon() { 
        count1++;  
        count2++;  
    }  
  
    public static SingleTon getInstance() {  
        return singleTon;  
    }  
}  
  
public class Demo {  
    public static void main(String[] args) {  
        SingleTon singleTon = SingleTon.getInstance(); 
        System.out.println("count1=" + singleTon.count1);  
        System.out.println("count2=" + singleTon.count2);  
    }  
}

执行的结果就是 count1=1 count2=2

因为这种情况下, 静态变量 count2比 singTon 先完成显示初始化,结果就是我们预料到的

查看详细解释 请参考 http://www.cnblogs.com/javaee6/p/3714716.html?utm_source=tuicool&utm_medium=referral

posted @ 2018-03-16 10:16  escapist  阅读(9297)  评论(0编辑  收藏