多态:如何理解父类引用指向子类对象

来源:互联网 发布:hp p2035n端口 编辑:程序博客网 时间:2024/06/04 23:31
之前一直对一个问题一直不理解,https://zhidao.baidu.com/question/1544175294520148587.html 里为什么要这么写Person p = new Student("张三其");
 而不是Student p = new Student("张三其");问了实习的leader,给了我这篇文章,http://www.cnblogs.com/xjhere/p/5842461.html (分割线后的文章)

第二篇原文地址:http://www.cnblogs.com/fickleness/archive/2013/06/21/3149011.html 
这一篇文章也对理解向上向下转型有帮助,向上转型,子类特有方法消失,override方法保留。父类引用不能指向子类(向下转型),编译会出错。

第三篇原文地址:https://zhidao.baidu.com/question/472310351.html
前两篇只说了向上转型和向下转型的特点,作用,但是并没有实际例子。第三篇算是一个案例吧。自己总结一下就是在函数的形式参数里用向上转型,也就是引用类型为父类比如person,那我调用方法的时候,我可以任意传入他的子类,这是一个向上转型吧。因为我可以任意传入他的子类,这就说明我能使用的范围非常广,如果我在形参里只定义一个比如student的引用,这样我就只能传入student类型。但是说好的多态呢?如果是形参为person,我可以通过C#中is或java中instanceof来判断是否可以向下转型,如果可以强制再向下转型,这样我在形参为person的这个函数里我就可以使用子类的不同方法。(个人理解请指正)

第四篇原文地址:http://www.cnblogs.com/wl0000-03/p/5815489.html

今天又发现了一个小例子,文章分为继承和多态两部分,多态部分作者给出了一个示例,算是第三篇例子的一个补充吧,如果理解了上面的内容,对第四篇应该轻车熟路。


稍微想了一下多态的使用,实习的公司做了很多机器人、战斗机这类的战斗机器。如果把机器人战斗机什么的都算作继承自mechine,mechine可以定义一个virtual battle的函数,robot和plane可以在各自的子类中override实现具体的战斗方式。比如如果要让战斗机器全部总攻击的时候可以把机器人全部放进一个List<mechine>类中,然后有两种选择,一,用is判断类型,接着强制转换,在调用子类的方法。二直接调用battle方法。因为父类的虚方法已经被子类给实现了,所以二方法应该比较简单。

-------------------------------------------------------------------------------------------------------

转载文章http://www.cnblogs.com/xjhere/p/5842461.html 

摘录

java多态,如何理解父类引用指向子类对象

要理解多态性,首先要知道什么是“向上转型”。

        我定义了一个子类Cat,它继承了Animal类,那么后者就是前者是父类。我可以通过   Cat c = new Cat(); 实例化一个Cat的对象,这个不难理解。

        但当我这样定义时:   Animal a = new Cat();  

表示定义了一个Animal类型的引用,指向新建的Cat类型的对象。由于Cat是继承自它的父类Animal,所以Animal类型的引用是可以指向Cat类型的对象的。

     那么这样做有什么意义呢?因为子类是对父类的一个改进和扩充,所以一般子类在功能上较父类更强大,属性较父类更独特,  定义一个父类类型的引用指向一个子类的对象既可以使用子类强大的功能,又可以抽取父类的共性。 

    所以,父类类型的引用可以调用父类中定义的所有属性和方法,而对于子类中定义而父类中没有的方法,它是无可奈何的;   同时,父类中的一个方法只有在父类中定义而在子类中没有重写的情况下,才可以被父类类型的引用调用;   对于父类中定义的方法,如果子类中重写了该方法,那么父类类型的引用将会调用子类中的这个方法,这就是动态连接。也可以叫做动态绑定。

     动态绑定是指”在执行期间(而非编译期间)“判断所引用对象的实际类型,根据实际的类型调用其相应的方法。

    看下面这段程序:

class Father {
    public void func1() 

   {
        func2();
    }

 // 这是父类中的func2()方法,因为下面的子类中重写了该方法 ,所以在父类类型的引用中调用时,这个方法将不再有效,取而代之的是将调用子类中重写的func2()方法
    public void func2() {
        System.out.println("AAA");
    }
}
class Child extends Father { // func1(int i)是对func1()方法的一个重载
   由于在父类中没有定义这个方法,所以它不能被父类类型的引用调用
    所以在下面的main方法中child.func1(68)是不对的
    public void func1(int i) {
        System.out.println("BBB");
    } // func2()重写了父类Father中的func2()方法   如果父类类型的引用中调用了func2()方法,那么必然是子类中重写的这个方法
    public void func2() {
        System.out.println("CCC");
    }
}
public class PolymorphismTest {
    public static void main(String[] args) {
        Father child = new Child();
        child.func1();// 打印结果将会是什么? } }

         上面的程序是个很典型的多态的例子。子类Child继承了父类Father,并重载了父类的func1()方法,重写了父类的func2()方法。重载后的 func1(int i)和func1()不再是同一个方法,由于父类中没有func1(int i),那么,父类类型的引用child就不能调用func1(int  i)方法。而子类重写了func2()方法,那么父类类型的引用child在调用该方法时将会调用子类中重写的func2()
    }
}

那么该程序将会打印出什么样的结果呢?       很显然,应该是“CCC”。     

    对于多态,可以总结它为:       

     一、使用父类类型的引用指向子类的对象;

    二、该引用只能调用父类中定义的方法和变量;

    三、如果子类中重写了父类中的一个方法,那么在调用这个方法的时候,将会调用子类中的这个方法;(动态连接、动态调用)

    四、变量不能被重写(覆盖),”重写“的概念只针对方法,如果在子类中”重写“了父类中的变量,那么在编译时会报错。

多态的3个必要条件:

        1.继承   2.重写   3.父类引用指向子类对象。

——————————————————————

向上转型: Person p = new Man() ; //向上转型不需要强制类型转化向下转型: Man man = (Man)new Person() ; //必须强制类型转化
原创粉丝点击