zoukankan      html  css  js  c++  java
  • java向上转型和向下转型

    转型是在继承的基础上而言的,继承是面向对象语言中,代码复用的一种机制,通过继承,子类可以复用父类的功能,如果父类不能满足当前子类的需求,则子类可以重写父类中的方法来加以扩展。

    向上转型:子类引用的对象转换为父类类型称为向上转型。通俗地说就是是将子类对象转为父类对象。此处父类对象可以是接口

    向下转型:父类引用的对象转换为子类类型称为向下转型。

    前者是一个向上转型,Animal dog 引用指向new Dog();子类对象当成父类对象,只能调用父类的成员,如果子类重写了父类的方法就根据这个引用指向调用子类重写的这个方法(这个方法就是覆盖override)。这个调用过程就称为“动态绑定”。

    转型需要注意的问题:

    向上转型时,父类指向子类引用对象会遗失除与父类对象共有的其他方法,也就是在转型过程中,子类的新有的方法都会遗失掉,在编译时,系统会提供找不到方法的错误。实例如下:

    public class Animal {

             public void eat(){

                       System.out.println("animal eatting...");

             }

    }

    class Bird extends Animal{

             public void eat(){

                       System.out.println("bird eatting...");

             }

             public void fly(){

                       System.out.println("bird flying...");

             }

    }

    class Main{      

             public static void main(String[] args) {

                       Animal b=new Bird(); //向上转型

                       b.eat();

             b.fly();  //此处提示在Animal中没有定义fly方法。

    在向下转型过程中,分为两种情况:

    情况一:如果父类引用的对象如果引用的是指向的子类对象,那么在向下转型的过程中是安全的。也就是编译是不会出错误的。

    情况二:如果父类引用的对象是父类本身,那么在向下转型的过程中是不安全的,编译不会出错,但是运行时会出现java.lang.ClassCastException错误。它可以使用instanceof来避免出错此类错误。实例如下:

    public class Girl {

             public void smile(){

                       System.out.println("girl smile()...");

             }

    }

    class MMGirl extends Girl{

                       @Override

             public void smile() {

                                         System.out.println("MMirl smile sounds sweet...");

             }

             public void c(){

                       System.out.println("MMirl c()...");

             }

    }

    class main{

                       public static void main(String[] args) {

                       Girl g1=new MMGirl(); //向上转型

                       g1.smile();

                       MMGirl mmg=(MMGirl)g1;    //向下转型,编译和运行皆不会出错

                       mmg.smile();

                       mmg.c();

                       Girl g2=new Girl();

               //MMGirl mmg1=(MMGirl)g2; //不安全的向下转型,编译无错但会运行会出错

               //mmg1.smile();

               //mmg1.c();

                       if(g2 instanceof MMGirl){

                                MMGirl mmg1=(MMGirl)g2;

                                mmg1.smile();

                                mmg1.c();

                       }

             }

    总结:

    1、父类引用可以指向子类对象,子类引用不能指向父类对象。

    2、把子类对象直接赋给父类引用叫upcasting向上转型,向上转型不用强制转型。

       如Father father = new Son();

    3、把指向子类对象的父类引用赋给子类引用叫向下转型(downcasting),要强制转型。

       如father就是一个指向子类对象的父类引用,把father赋给子类引用son 即Son son =(Son)father;

       其中father前面的(Son)必须添加,进行强制转换。

    4、upcasting 会丢失子类特有的方法,但是子类overriding 父类的方法,子类方法有效

    5、向上转型的作用,减少重复代码,父类为参数,调有时用子类作为参数,就是利用了向上转型。这样使代码变得简洁。体现了JAVA的抽象编程思想。

  • 相关阅读:
    angular6跨域问题proxy.conf.json
    angular2 防止刷新页面 参数丢失
    自定义修改table样式 scoped会影响
    angular ngfor和ngif指令共用
    yum 是轮船发动机,brew 是汽车发动机,不能混用。
    Vue watch 监听 computed
    vue 路由跳转,参数消失问题
    vue 使用:class切换高亮 点击路由跳转 上个组件的点击事件保存的参数 在下一次重复进入这个组件的时候 默认值都已经还原了 得通过路由跳转的时候 把上个组件的状态通过路由保存下来 通过下一次进入这个组件的时候 获取路由 渲染当前页面的:class 进行高亮
    函数防抖 函数节流
    vue router-view router-link RouterView【命令视图】和RouterLink【命令路线】本身是两个组件。 命名路由
  • 原文地址:https://www.cnblogs.com/fickleness/p/3149011.html
Copyright © 2011-2022 走看看