今天在百度知道上看见某人问的一个问题,改装了一下,给人的感觉会更加诡异。下面来看看这个小程序:
class
TestA {
int x = 1 ;
public TestA() {
a1();
}
public void a1() {
System.out.println( " x in TestA is " + x);
}
}
public class TestB extends TestA {
int x = 10 ;
public TestB() {
x = 1000 ;
}
public void a1() {
System.out.println( " x in TestB is " + x);
}
public static void main(String[] args) {
new TestB();
}
}
int x = 1 ;
public TestA() {
a1();
}
public void a1() {
System.out.println( " x in TestA is " + x);
}
}
public class TestB extends TestA {
int x = 10 ;
public TestB() {
x = 1000 ;
}
public void a1() {
System.out.println( " x in TestB is " + x);
}
public static void main(String[] args) {
new TestB();
}
}
现在来猜猜运行结果会是什么?TestA还是TestB?x是1,10还是100?
如果你对上面几个数字念念不忘,那么恭喜你,你提前知道了自己错了。正确的答案是:
x in testB is 0
下面来说说原因:
类初始化时构造函数调用顺序:
(1)初始化对象的存储空间为零或null值;
(2)调用父类构造函数;
(3)按顺序分别调用类成员变量和实例成员变量的初始化表达式;
(4)调用本身构造函数。
首先new TestB()实例化一个TestB的对象
(1)先给自己分配了空间,并赋初始值;
(2)由于TestB继承至TestA,这里使用的是默认的没有参数的构造函数,它会先调用父类的构造函数,父类的构造函数中调用了a1()方法,但我们实例化的是一个子类TestB的对象,在子类中已经覆盖了a1()方法,因此调用的是子类的a1()方法,这时候步骤(3)还并没有被执行,因此x还是初始值,int的默认初始值是0;
(3)执行x=10;
(4)执行TestB中的构造函数中的x=100;
我们的程序是在第(2)步中打印结果的,所以结果会是上面那样。