《疯狂JAVA讲义》——多态

Java引用变量有两个类型:一个是编译时的类型,一个是运行时的类型。

编译时的类型由声明该变量时所使用的类型决定;
运行时的类型由实际赋值给该变量的对象决定。

当编译时的类型与运行时的类型不一致时,会发生多态(Polymorphism)。

代码:

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

class SubClass extends BaseClass
{
    //重新定义一个book实例Field隐藏父类的book实例Field
    public String book = "疯狂java讲义";
    public void test()
    {
        System.out.println("子类覆盖父类的方法");
    }
    public void sub()
    {
        System.out.println("子类的普通方法");
    }
}

public class PolymorphismTest
{
    public static void main(String[] args)
    {
        //编译时和运行时类型一样,不存在多态
        BaseClass bc = new BaseClass();
        System.out.println(bc.book);//输出6
        bc.base();//输出“父类的普通方法”
        bc.test();//输出“父类被覆盖的方法”

        //编译时和运行时类型一样,不存在多态
        SubClass sc = new SubClass();
        System.out.println(sc.book);//输出“疯狂java讲义”
        sc.base();//输出“父类的普通方法”,执行子类从父类继承的base方法
        sc.test();//输出“子类覆盖基类的方法”
        sc.sub();//输出“子类的普通方法”

        //编译时和运行时类型不一样,多态发生            
        BaseClass polymorphicBc = new SubClass();
        System.out.println(polymorphicBc.book);//输出6,表明访问的是父类的Field
        polymorphicBc.base();//输出“父类的普通方法”,执行子类从父类继承到的base方法
        polymorphicBc.test();//输出“子类覆盖基类的方法”,执行子类的test方法
        polymorphicBc.sub();//编译报错,BaseClass类没有sub方法
    }
}

Java允许把一个子类对象直接赋给一个父类引用,即父类引用可以指向子类对象,无需任何类型转换,被称为向上转型(upcasting),由系统自动完成。

代码中的polymorphicBc对象在编译时是BaseClass类型,而在运行时是SubClass类型。所以polymorphicBc.sub();语句在编译时会引发错误,因为BaseClass类没有sub方法。

通过引用变量来访问其包含的Field实例时,系统总是试图访问它编译时类型所定义的Field,而不是它运行时所定义的Field。所以System.out.println(polymorphicBc.book);输出6,即访问的是BaseClass类中定义的int类型的book。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值