一 多态介绍
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类并不存在此方法