注:该篇文章已与我的个人博客同步更新。欢迎移步https://cqh-i.github.io/体验更好的阅读效果。
请写出如下代码运行后的结果
class A {
public A() {
PrintFields();
}
public void PrintFields() {
}
}
class B extends A {
int x = 1;
int y;
public B() {
y = -1;
}
public void PrintFields() {
System.out.println("x=" + x + ",y=" + y);
}
}
public class Test {
public static void main(String[] args) {
B b = new B();
}
}
结果:
x=0,y=0
解析:
因为在创建B类对象的时候,调用了B类构造器,但在调用构造器之前,JAVA先对B类内各成员变量进行初始化,在这个过程中,首先完成对基类的成员初始化然后才是B类自己的成员初始化,初始化过程中,无论你在成员变量处指定任何值,JAVA都会赋给各成员变量默认的初始值,对于数值类型就是0,对于引用就是NULL,所以初始化完毕,x=0,y=0。
然后再调用基类的构造器,但在A类的构造器中调用了printFields(),而B类也有一个printFields(),
它覆盖了A类的printFields(),由于多态的原因,所以JAVA调用B类的printFields(),所以输出x=0,y=0。
在从A类构造器返回到B类构造器的时候,情况有些改变,x值此时由0变成1,说明你在定义变量x时所赋的初值在此时起了作用,再次输出x=1,y=0。修改后的程序如下,只是在B的构造函数里增加了一条输出语句
class A {
public A() {
PrintFields();
}
public void PrintFields() {
}
}
class B extends A {
int x = 1;
int y;
public B() {
System.out.println("x=" + x + ",y=" + y);
y = -1;
}
public void PrintFields() {
System.out.println("x=" + x + ",y=" + y);
}
}
public class Test {
public static void main(String[] args) {
B b = new B();
}
}
运行结果:
x=0,y=0
x=1,y=0