注:该篇文章已与我的个人博客同步更新。欢迎移步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
Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐