一个值得深思的JVM示例

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;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值