日期:2014-05-20  浏览次数:20762 次

一道经典的JAVA面试题
public class M {

	public static void main(String[] args) {

		N n = N.getInstence();
		System.out.println("a = " + n.a);
		System.out.println("b = " + n.b);
	}

}

class N {

	public static int a;
	public static int b = 0;
	private static N n = new N();

	private N() {
		a++;
		b++;
	}

	public static N getInstence() {
		return n;
	}
}

????? 上面的代码普遍都能知道答案,输出是:?

a = 1
b = 1

?????? 如果把上面的N类里面的创建N的成员对象的时候调换位置,代码如下:

public class M {

	public static void main(String[] args) {

		N n = N.getInstence();
		System.out.println("a = " + n.a);
		System.out.println("b = " + n.b);
	}

}

class N {
	
	private static N n = new N();
	public static int a;
	public static int b = 0;

	private N() {
		a++;
		b++;
	}

	public static N getInstence() {
		return n;
	}
}

????? 现在输出的确不在是上面的结果,结果如下:

a = 1
b = 0

?????? 我们都知道类加载的时候首先加载的是静态代码,在第一个代码里面,当执行 N n = N.getInstence();的时候,java虚拟机会去加载N类,在加载N类的时候,按照顺序:

public static int a;
public static int b = 0;
private static N n = new N();

?????? 首先给静态变量分配内存空间,其次给静态变量赋予默认值然得到 a = 0;b = 0;n = null;然后显示初始化各静态变量,因为a没有显示赋初值,所以a = 0,而b显示赋初值为0,所以b =?,然后显示为n赋初值,在new N()的时候对a、b自增,得到 a = 1;b = 1;所以最终结果如上述。?

?????? 在第二段代码里面,当执行 N n = N.getInstence();的时候,java虚拟机会去加载N类,在加载N类的时候,按照顺序:

private static N n = new N();
public static int a;
public static int b = 0;

??????? 首先给静态变量分配内存空间,其次给静态变量赋予默认值,得到 n = null ;a = 0; b = 0;

? 然后显示初始化各静态变量,n为N对象,在new N()的时候对a、b自增,得到a = 1;b = 1;

? 因为a没有显示赋予初值,所以a还是为a = 1;而b显示赋予初值为0,所以b = 0;得到最终结果如上述。

?????? 所以得到最终的结论是,jvm在加载类的时候对静态代码按如下步骤处理:

??????? 1、当jvm在加载类的时候,首先是对各个静态变量分配空间。

??????? 2、对各个静态变量赋予默认值(比如int、long型的默认值为0,Object的默认值为null)。

??????? 3、对各个静态变量按顺序显示初始化(即按照代码里面的初始化为其初始化)。