Java多态性

一 多态介绍

Java引用变量有两种类型:一个是编译时类型,一个是运行时类型,编译时类型由声明该变量时使用的类型来决定,运行时类型由实际赋给该变量的对象决定,如果编译时的类型和运行时的类型不一致,就可能出现多态。

二多态性应用

1 代码示例

class BaseClass
{
	public int book = 6;
	public void base()
	{
		System.out.println("父类的普通方法");
	}
	public void test()
	{
		System.out.println("父类的被覆盖的方法");
	}
}
public class TestsubClass extends BaseClass
{
	//重新定义一个book实例变量隐藏父类的book实例变量
	public String book = "软件秘笈";
	public void test()
	{
		System.out.println("子类的覆盖父类的方法");
	}
	public void sub()
	{
		System.out.println("子类的普通方法");
	}
	public static void main(String[] args)
	{
		// 下面编译时类型和运行时类型完全一样,因此不存在多态
		BaseClass bc = new BaseClass();
		// 输出 6
		System.out.println(bc.book);
		// 下面两次调用将执行BaseClass的方法
		bc.base();
		bc.test();
		// 下面编译时类型和运行时类型完全一样,因此不存在多态
		TestsubClass sc = new TestsubClass();
		// 输出"轻量级Java EE企业应用实战"
		System.out.println(sc.book);
		// 下面调用将执行从父类继承到的base()方法
		sc.base();
		// 下面调用将执行从当前类的test()方法
		sc.test();
		// 下面编译时类型和运行时类型不一样,多态发生
		BaseClass ploymophicBc = new TestsubClass();
		// 输出6 —— 表明访问的是父类对象的实例变量
		System.out.println(ploymophicBc.book);   //a
		// 下面调用将执行从父类继承到的base()方法
		ploymophicBc.base();  //b
		// 下面调用将执行从当前类的test()方法
		ploymophicBc.test();   //c
		// 因为ploymophicBc的编译类型是BaseClass,
		// BaseClass类没有提供sub方法,所以下面代码编译时会出现错误。
		// ploymophicBc.sub();  //d
	}
}  

2 运行结果

6

父类的普通方法

父类的被覆盖的方法

软件秘笈

父类的普通方法

子类的覆盖父类的方法

6

父类的普通方法

子类的覆盖父类的方法

3 结果分析

3.1代码声明了3个引用变量bc,sc和ploymophicBc,针对前两个引用不涉及多态,运行完全正常,在此不作讨论,仅分析第3个引用变量,编译时和运行时类型不一致

3.2a处代码访问的父类对象的实例变量,说明对象的实例变量不具备多态性,系统总是试图访问它编译时类型所定义的成员变量,而不是它运行时类型所定义的成员变量

3.3b处代码调用父类的base方法,因为子类并没重写此方法

3.4c处代码调用孑类的test方法,因为子类重写了此方法,引用变量运行时执行它运行时类型所具有的方法

3.5d处代码如果不注释,则无法编译通过,因为引用变量在编译时只能调用其编泽时类型所具有的方法,但编译时BaseC1ass类并不存在此方法

猜你喜欢

转载自cakin24.iteye.com/blog/2327956