你真的理解了继承和多态吗


public class DoYouReallyUnderstandPolymorphism {

public static void main(String[] args) {
A a = new A();
B b = new B();
a.s = "[AA]";
b.s = "[BB]";
System.out.println(a.s); // prints "[AA]"
System.out.println(b.s); // prints "[BB]"
System.out.println(a.getS()); // prints "[AA]"
System.out.println(b.getS()); // prints "[BB]"
System.out.println("====================");
a = b; // a now refers to object b
System.out.println(a.s); // prints "[A]" <<--1-- the class A copy
System.out.println(b.s); // prints "[BB]"
System.out.println(a.getS()); // prints "[BB]"
System.out.println(b.getS()); // prints "[BB]"
System.out.println("====================");
((A)b).s = "[AA]"; // <<--2-- changes the class A copy in object b
System.out.println(a.s); // prints "[AA]" <<--3-- class A copy changed
System.out.println(b.s); // prints "[BB]"
System.out.println(a.getS()); // prints "[BB]"
System.out.println(b.getS()); // prints "[BB]"
}

}

class A {
String s = "[A]";
String getS() {
return s;
}
}

class B extends A{
String s = "";
String getS() {
return s;
}
}

解释:
  直观的讲,我们很容易轻信当"a = b;"以后,变量a指向的对象是B类的b那个对象,自然a.s就应该等同于b.s,然而事实并非如此。当B继承A时,父类A的字段s并没有被B的字段s取代,而是保留了一份拷贝,所谓重写(Override),那是对方法而言的。于是,当我们new B()时,在实际创建的对象中,包含了两个版本的字段s,一个"[A]"(属于A类)一个""(属于B类)。而方法getS()只有一个版本。这就是在继承过程中字段和方法的区别。也就是说,重写的概念和字段无关。在第16行,我们通过a.s访问的是b这个对象中保留的A类的字段s;而在21行,我们改变的正是这个A类版本的s字段。

  多态的精髓在于动态确定对象的行为,而对象的行为体现在方法而非字段,字段代表的更多的是对象的状态。于是只有方法的多态而没有字段的多态。从上面的代码可以看出,不管你用什么类型的变量存放对象b的引用,最终调用的方法版本都是b的真实类型那个版本,这就是多态的威力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值