放假呆在家没事,顺手把think in java拿起来看,对于这个大部头我一直都木有坚持从头往后翻完,而是木事的时候就拿起来翻翻,经常都学到了新的东西!
多态,从刚开始学习java的时候就开始了解的,自己也经常使用,但是看看下面一段代码
package thinkInJava.polymorphism;
public class FieldAccess
{
public static void main(String[] args)
{
Super sup = new Sub();
System.out.println("sup.field=" + sup.field);
System.out.println("sup.getField()=" + sup.getField());
}
}
class Super
{
public int field = 0;
public int getField()
{
return field;
}
}
class Sub extends Super
{
public int field = 1;
public int getField()
{
return field;
}
public int getSuperField()
{
return super.field;
}
}
输出的结果不是我以为的
sup.field=1
sup.getField()=1
而是
sup.field=0
sup.getField()=1
这是为么子呢?原来多态是针对于普通方法的调用来说的,如果直接访问某个域,这个访问将在编译期进行解析,当Sub对象转型为Super引用的时候,任何域的访问操作都是由编译器来解析的而不是多态的,所以sup.field访问的是Super类的域。虽然很容易搞混,但是平常我们一般都把域设置为private,所以这种假设的情况是不会发生的。
package thinkInJava.polymorphism;
public class StaticPolymorphism
{
public static void main(String[] args)
{
StaticSuper sup = new StaticSub();
sup.dynamicGet();
sup.staticGet();
}
}
class StaticSuper
{
public static void staticGet()
{
System.out.println("Base staticGet()");
}
public void dynamicGet()
{
System.out.println("Base dynamicGet()");
}
}
class StaticSub extends StaticSuper
{
public static void staticGet()
{
System.out.println("Derived staticGet()");
}
public void dynamicGet()
{
System.out.println("Derived dynamicGet()");
}
}
上面程序的结果又和我以为的不一样,结果为:
Derived dynamicGet()
Base staticGet()
package thinkInJava.polymorphism;
public class PrivateOverride
{
private void f()
{
System.out.println("private f()");
}
public static void main(String[] args)
{
PrivateOverride po = new Derived();
po.f();
}
}
class Derived extends PrivateOverride
{
public void f()
{
System.out.println("public f()");
}
}
这个结果又是什么呢?结果是private f()。private方法会被认为是final方法,对导出类是屏蔽的,更不用说对其进行override,Derived类中的f()方法是一个全新的方法,貌似这很容易引起迷惑,所以取名字的时候注意,导出类中的方法名最好不要和父类中的private方法重名,要不然可能会产生误解。