嗯,继承的一个小知识点,前段时间线上碰到的一个问题,不说废话,直接上一个demo看下吧
class Parent{
public int value = 5;
public int getValue(){
return value;
}
}
class Child extends Parent{
public int value = 10;
public static void main(String[] args) {
Parent parent = new Child();
Child child = new Child();
System.out.println(parent.value);
System.out.println(parent.getValue());
System.out.println(child.value);
System.out.println(child.getValue());
}
}
结果:
5
5
10
5
之前一直以为,子类继承父类,同名变量的值只有一个,即会覆盖,因为想着对象实例化后,在内存中,这个对象的同名成员变量只有那么一个,但这个demo告诉我,其实是两个:虽然同名,但是他们的“域”不同,现在还不知道这个所谓“域”怎么解释,但照着结果来看,可以这么理解:内存中对象数据分为了两块(或者多块,这取决于继承链多长),一块是父类属性,一块是子类自己属性,在我们看来同名变量,但因为所在域不同,他们并不是同一个(同名方法亦然,但又有不同,这个后面再说)。
如果有了以上的理解,那么就能比较容易的知道demo的答案从何而来:同名变量value,是何值,取决于你如何获取的,即这个value真正指向的是哪个,在demo中:
parent.value,很显然,parent是父类静态引用,指向的是父类的value
parent.getValue(),getValue()是父类方法,返回的是父类的value
child.value,child是子类的静态引用,指向的是child的value
child.getValue(),getValue()是父类方法,返回的是父类的value,如果这个时候子类有覆盖getValue(),且返回的是子类的value,即:
class Child extends Parent{
public int value = 10;
public int getValue(){
return value;
}
}
那么child.getValue()返回的是子类的value,即10。(注1)
但如果,覆盖是:
class Child extends Parent{
public int value = 10;
public int getValue(){
return super.value;
}
}
返回的则是父类的value,即为5。
好了,说了变量,再说下方法,同名方法,在注1中,这个时候parent.getValue()返回的也是10,为什么呢?因为方法是跟着动态类型走的,而不是静态类型,parent的这个对象,静态类型是Parent,动态类型是Child,所以这个时候调用的是子类的getValue()方法,而不是父类的,如果Parent parent = new Parent(),则parent.getValue()返回的就是5了。