引用:https://www.cnblogs.com/smallFishs/p/6185627.html
第一次写文章,而且我也是刚刚知道这个问题的原因,还是我老师给我解释的,写出来一方面是为了加深记忆方便理解,另一方面希望有更多人知道这个原因,下面看一下这个问题。
看一下这段代码,大家觉得执行结果是多少:是5 10 5 10 吗?
第一个代码
![](https://img.haomeiwen.com/i10608194/537d8a5865d3bf00.png)
![](https://img.haomeiwen.com/i10608194/afe79db8c03fa638.png)
结果是5 0 5 0.
同样,我把上面代码改变一下顺序。
修改后的代码
![](https://img.haomeiwen.com/i10608194/188d958085c77bfc.png)
![](https://img.haomeiwen.com/i10608194/78be7b05d744fd9a.png)
只是把static A a放到了后面,他的结果就变成了5 10 5 10。大家应该知道有些门道了,但是还是不太清楚对吧。
下面介绍一下jvm类加载器的执行过程:
![](https://img.haomeiwen.com/i10608194/fd5ebad7bb627c2a.jpg)
对应上面的例子,讲解一下过程吧:
1.在准备阶段,进行static变量申请空间,对应的就是c1,c2,a变量申请内存空间,但是此时还没有进行初始化、赋值操作。
2.在初始化阶段。就是我们A类中进行的操作,对static变量的初始化就在这个阶段。但是要注意,我们一开始写的是
public static A a = new A();这个初始化包含有对象初始化,这个是对应着第三个阶段:对象实例化阶段。也就是说,在类初始化(ClInit)未完成结束之前,先执行了对象实例化操作。在上例中,先执行了A的构造方法,分别对c1,c2赋值后,再对c2进行了初始化(这个地方我也不清楚应该叫他赋值还是初始化了,因为先执行了一次赋值)。所以c1为5,c2为0;
针对这个问题,
1.我们可以参考刚才修改后的代码:我们将有关a对象的初始化操作放在最后,确保初始化阶段(Init)完成之后,在进行对象初始化操作。
2.
当然这个代码的饿汉单例模式还不完善,还有很多种实现方法,我们只是针对上述实现方式产生的问题作出解决方案。
网友评论