Java中类的加载顺序剖析(常用于面试题)
这其实是去年校招时我遇到的一道阿里巴巴的笔试题(承认有点久远了-。-),嗯,如果我没记错的话,当时是作为java方向的一道选做大题。当然题意没有这么直白,题目只要求你写出程序运行后所有system.out.println的输出结果,其中程序是题目给的,而各个system.out.println的执行顺序不同会导致最后程序输出的结果也不同。
具体的题目我肯定记不清,不过我们可以换个直接的问法,如果类a和类b中有静态变量,静态语句块,非静态变量,非静态语句块,构造函数,静态方法,非静态方法,同时类a继承类b,请问当实例化a时,类内部的加载顺序是什么?
当时我也是一头雾水,事后我就自己写了一个小demo,这才知道了类内部的实际加载顺,测试代码如下:
class b:
public class b{ //静态变量 static int i=1; //静态语句块 static { system.out.println("class b1:static blocks"+i); } //非静态变量 int j=1; //静态语句块 static{ i++; system.out.println("class b2:static blocks"+i); } //构造函数 public b(){ i++; j++; system.out.println("constructor b: "+"i="+i+",j="+j); } //非静态语句块 { i++; j++; system.out.println("class b:common blocks"+"i="+i+",j="+j); } //非静态方法 public void bdisplay(){ i++; system.out.println("class b:static void bdisplay(): "+"i="+i+",j="+j); return ; } //静态方法 public static void btest(){ i++; system.out.println("class b:static void btest(): "+"i="+i); return ; } }
class a:
public class a extends b{ //静态变量 static int i=1; //静态语句块 static { system.out.println("class a1:static blocks"+i); } //非静态变量 int j=1; //静态语句块 static{ i++; system.out.println("class a2:static blocks"+i); } //构造函数 public a(){ super(); i++; j++; system.out.println("constructor a: "+"i="+i+",j="+j); } //非静态语句块 { i++; j++; system.out.println("class a:common blocks"+"i="+i+",j="+j); } //非静态方法 public void adisplay(){ i++; system.out.println("class a:static void adisplay(): "+"i="+i+",j="+j); return ; } //静态方法 public static void atest(){ i++; system.out.println("class a:static void atest(): "+"i="+i); return ; } }
class classloading :
public class classloading { public static void main (string args[]) { a a=new a(); a.adisplay(); } }
程序运行结果如图:
通过上述示图,我们可以比较清晰的看出java类的整个加载过程。
1.若要加载类a,则先加载执行其父类b(object)的静态变量以及静态语句块(执行先后顺序按排列的先后顺序)。
2.然后再加载执行类a的静态变量以及静态语句块。(并且1、2步骤只会执行1次)
3.若需实例化类a,则先调用其父类b的构造函数,并且在调用其父类b的构造函数前,依次先调用父类b中的非静态变量及非静态语句块.最后再调用父类b中的构造函数初始化。
4.然后再依次调用类a中的非静态变量及非静态语句块.最后调用a中的构造函数初始化。( 并且3、4步骤可以重复执行)
5.而对于静态方法和非静态方法都是被动调用,即系统不会自动调用执行,所以用户没有调用时都不执行,主要区别在于静态方法可以直接用类名直接调用(实例化对象也可以),而非静态方法只能先实例化对象后才能调用。
ok,今天就总结到这里了,如果有地方说的不好或有错误的地方,欢迎大家指出,定当改正,谢谢。