Java中的多态

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

注:实例变量不具备多态,实例方法才具备多态

/**
 * 说明:理解多态,实例变量不具多态性,实例方法才具备多态性
 * @author LiuYP_1024
 *
 */


class BaseClass{
	public int sign=1024;
	public void base() {
		System.out.println("父类的普通方法");
	}
	public void test() {
		System.out.println("父类被覆盖的方法");
	}
}

public class SubClass extends BaseClass{
	//重新定义一个sign实例变量隐藏父类的sign实例变量
	public String sign="子类的String变量";
	public void test() {
		System.out.println("子类的覆盖父类的方法");
	}
	public void sub() {
		System.out.println("子类的普通方法");
	}
			
	public static void main(String[] args) {
		//下面编译时类型与运行时类型完全一样,所以不存在多态
		BaseClass bc=new BaseClass();
		System.out.println(bc.sign);
		bc.base();
		bc.test();
		
		//下面编译时类型与运行时类型完全一样,所以不存在多态
		SubClass sc=new SubClass();
		System.out.println(sc.sign);
		sc.base();//执行从父类继承到的base()方法
		sc.test();//执行当前类继承到的test()方法
		
		//下面编译时类型与运行类型不一样,所以存在多态
		BaseClass baseandsub=new SubClass();
		//注意下面一句
		System.out.println(baseandsub.sign);//实例变量不具备多态性,所以输出的时父类的实例变量sign=1024
		baseandsub.base();//执行从父类继承到的base()方法
		baseandsub.test();//执行当前类继承到的test()方法
		
		//注意下面一句,因为编译时类型是BaseClass,该类没有提供sub()方法,因此下面代码出错
		//baseandsub.sub();
	}
}

运行结果如下:

1024
父类的普通方法
父类被覆盖的方法
子类的String变量
父类的普通方法
子类的覆盖父类的方法
1024
父类的普通方法
子类的覆盖父类的方法

上面程序的main()方法中显式创建了三个引用变量,对于前两个引用变量bc和sc,它们编译时类型和运行时类型完全相同,因此调用它们的成员变量和方法非常正常,完全没有任何问题。但第三个引用变量baseandsub则比较特殊,它的编译时类型是BaseClass,而运行时类型是SubClass,当调用该引用变量的test0方法(BaseClas类中定义了该方法,子类Subclass覆盖了父类的该方法)时,实际执行的是SubClass类中覆盖后的test)方法,这就可能出现多态了。
因为子类其实是一种特殊的父类,因此Java允许把一个子类对象直接赋给一个父类引用变量,无须任何类型转换,或者被称为向上转型(upcasting),向上转型由系统自动完成。
当把一个子类对象直接赋给父类引用变量时,例如上面的Baseclass baseandsub=new SubClass();这个baseandsub引用变量的编译时类型为BaseClass,而运行时类型为SubClass,当运行时调用该引用变量的方法时,其方法行为总是表现出子类方法的行为特征,而不是父类方法的行为特征,这就可能出现:相同类型的变量、调用同一个方法时呈现出多种不同的行为特征,这就是多态。

上面的main()方法中注释了baseandsub.sub(),这行代码在编译时会引发错误。虽然baseandsub引用变量实际上确实包含sub()方法,但因为它的编译时类型为BaseClass,因此编译时无法调用sub)方法。
与方法不同的是,对象的实例变量则不具备多态性。比如上面的baseandsub引用变量,程序输出它的sign实例变重时,并不是输出SubClasS类重定义的实例变量,而是输出Baseclass类的实例变量。
引用变量在编译阶段只能调用其编译时类型所具有的方法,但运行时则执行它运行时类型所具有的方法。因此,编写Java代码时,引用变量只能调用声明该变量时所用类里包含的方法。例如,通过Object p=new Person)代码定义一个变量p,则这个p只能调用Obiect类的方法,而不能调用Person类里定义的方法。

 

内容整理来源于:疯狂Java讲义

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值