多态中的域以及静态方法

放假呆在家没事,顺手把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方法重名,要不然可能会产生误解。

经验分享 程序员 微信小程序 职场和发展