• java基础(三)-----java的三大特性之多态


    本文参考自:java基础(三)-----java的三大特性之多态

    正文

      面向对象编程有三大特性:封装、继承、多态。

           封装隐藏了类的内部实现机制,可以在不影响使用的情况下改变类的内部结构,同时也保护了数据。对外界而已它的内部细节是隐藏的,暴露给外界的只是它的访问方法。

           继承是为了重用父类代码。两个类若存在IS-A的关系就可以使用继承。同时继承也为实现多态做了铺垫。那么什么是多态呢?多态的实现机制又是什么?请看我一一为你揭开。

    多态

      所谓多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。因为在程序运行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态,这就是多态性。

           比如你是一个酒神,对酒情有独钟。某日回家发现桌上有几个杯子里面都装了白酒,从外面看我们是不可能知道这是些什么酒,只有喝了之后才能够猜出来是何种酒。你一喝,这是剑南春、再喝这是五粮液、再喝这是酒鬼酒….在这里我们可以描述成如下:

           酒 a = 剑南春

           酒 b = 五粮液

           酒 c = 酒鬼酒

           …

           这里所表现的的就是多态。剑南春、五粮液、酒鬼酒都是酒的子类,我们只是通过酒这一个父类就能够引用不同的子类,这就是多态——我们只有在运行的时候才会知道引用变量所指向的具体实例对象。

           诚然,要理解多态我们就必须要明白什么是“向上转型”。在继承中我们简单介绍了向上转型,这里就在啰嗦下:在上面的喝酒例子中,酒(Win)是父类,剑南春(JNC)、五粮液(WLY)、酒鬼酒(JGJ)是子类。我们定义如下代码:

           JNC a = new JNC();

           对于这个代码我们非常容易理解无非就是实例化了一个剑南春的对象嘛!但是这样呢?

           Wine a = new JNC();

           在这里我们这样理解,这里定义了一个Wine 类型的a,它指向JNC对象实例。由于JNC是继承与Wine,所以JNC可以自动向上转型为Wine,所以a是可以指向JNC实例对象的。这样做存在一个非常大的好处,在继承中我们知道子类是父类的扩展,它可以提供比父类更加强大的功能,如果我们定义了一个指向子类的父类引用类型,那么它除了能够引用父类的共性外,还可以使用子类强大的功能。

           但是向上转型存在一些缺憾,那就是它必定会导致一些方法和属性的丢失,而导致我们不能够获取它们。所以父类类型的引用可以调用父类中定义的所有属性和方法,对于只存在与子类中的方法和属性它就望尘莫及了---1。

    复制代码
     1 public class Wine {
     2     public void fun1(){
     3         System.out.println("Wine 的Fun.....");
     4         fun2();
     5     }
     6     
     7     public void fun2(){
     8         System.out.println("Wine 的Fun2...");
     9     }
    10 }
    11  
    12 public class JNC extends Wine{
    13     /**
    14      * @desc 子类重写父类方法
    15      *        父类中不存在该方法,向上转型后,父类是不能引用该方法的
    16      * @param a
    17      * @return void
    18      */
    19     public void fun1(String a){
    20         System.out.println("JNC 的 Fun1...");
    21         fun2();
    22     }
    23     
    24     /**
    25      * 子类重写父类方法
    26      * 指向子类的父类引用调用fun2时,必定是调用该方法
    27      */
    28     public void fun2(){
    29         System.out.println("JNC 的Fun2...");
    30     }
    31 }
    32  
    33 public class Test {
    34     public static void main(String[] args) {
    35         Wine a = new JNC();
    36         a.fun1();
    37     }
    38 }
    39 -------------------------------------------------
    40 Output:
    41 Wine 的Fun.....
    42 JNC 的Fun2...
    复制代码

    从程序的运行结果中我们发现,a.fun1()首先是运行父类Wine中的fun1().然后再运行子类JNC中的fun2()。

          分析:在这个程序中子类JNC重载了父类Wine的方法fun1(),重写fun2(),而且重载后的fun1(String a)与 fun1()不是同一个方法,由于父类中没有该方法,向上转型后会丢失该方法,所以执行JNC的Wine类型引用是不能引用fun1(String a)方法。而子类JNC重写了fun2() ,那么指向JNC的Wine引用会调用JNC中fun2()方法。

          所以对于多态我们可以总结如下:

          指向子类的父类引用由于向上转型了,它只能访问父类中拥有的方法和属性,而对于子类中存在而父类中不存在的方法,该引用是不能使用的,尽管是重载该方法。若子类重写了父类中的某些方法,在调用该些方法的时候,必定是使用子类中定义的这些方法(动态连接、动态调用)。

    多态的实现

    实现条件

          在刚刚开始就提到了继承在为多态的实现做了准备。子类Child继承父类Father,我们可以编写一个指向子类的父类类型引用,该引用既可以处理父类Father对象,也可以处理子类Child对象,当相同的消息发送给子类或者父类对象时,该对象就会根据自己所属的引用而执行不同的行为,这就是多态。即多态性就是相同的消息使得不同的类做出不同的响应。

          Java实现多态有三个必要条件:继承、重写、向上转型。

             继承:在多态中必须存在有继承关系的子类和父类。

             重写:子类对父类中某些方法进行重新定义,在调用这些方法时就会调用子类的方法。

             向上转型:在多态中需要将子类的引用赋给父类对象,只有这样该引用才能够具备技能调用父类的方法和子类的方法。

          只有满足了上述三个条件,我们才能够在同一个继承结构中使用统一的逻辑实现代码处理不同的对象,从而达到执行不同的行为。

          对于Java而言,它多态的实现机制遵循一个原则:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。

    实现形式

     在Java中有两种形式可以实现多态。继承和接口。

    基于继承实现的多态

          基于继承的实现机制主要表现在父类和继承该父类的一个或多个子类对某些方法的重写,多个子类对同一方法的重写可以表现出不同的行为。

    复制代码
     1 public class Wine {
     2     private String name;
     3     
     4     public String getName() {
     5         return name;
     6     }
     7  
     8     public void setName(String name) {
     9         this.name = name;
    10     }
    11  
    12     public Wine(){
    13     }
    14     
    15     public String drink(){
    16         return "喝的是 " + getName();
    17     }
    18     
    19     /**
    20      * 重写toString()
    21      */
    22     public String toString(){
    23         return null;
    24     }
    25 }
    26  
    27 public class JNC extends Wine{
    28     public JNC(){
    29         setName("JNC");
    30     }
    31     
    32     /**
    33      * 重写父类方法,实现多态
    34      */
    35     public String drink(){
    36         return "喝的是 " + getName();
    37     }
    38     
    39     /**
    40      * 重写toString()
    41      */
    42     public String toString(){
    43         return "Wine : " + getName();
    44     }
    45 }
    46  
    47 public class JGJ extends Wine{
    48     public JGJ(){
    49         setName("JGJ");
    50     }
    51     
    52     /**
    53      * 重写父类方法,实现多态
    54      */
    55     public String drink(){
    56         return "喝的是 " + getName();
    57     }
    58     
    59     /**
    60      * 重写toString()
    61      */
    62     public String toString(){
    63         return "Wine : " + getName();
    64     }
    65 }
    66  
    67 public class Test {
    68     public static void main(String[] args) {
    69         //定义父类数组
    70         Wine[] wines = new Wine[2];
    71         //定义两个子类
    72         JNC jnc = new JNC();
    73         JGJ jgj = new JGJ();
    74         
    75         //父类引用子类对象
    76         wines[0] = jnc;
    77         wines[1] = jgj;
    78         
    79         for(int i = 0 ; i < 2 ; i++){
    80             System.out.println(wines[i].toString() + "--" + wines[i].drink());
    81         }
    82         System.out.println("-------------------------------");
    83  
    84     }
    85 }
    86 OUTPUT:
    87 Wine : JNC--喝的是 JNC
    88 Wine : JGJ--喝的是 JGJ
    89 -------------------------------
    复制代码

          在上面的代码中JNC、JGJ继承Wine,并且重写了drink()、toString()方法,程序运行结果是调用子类中方法,输出JNC、JGJ的名称,这就是多态的表现。不同的对象可以执行相同的行为,但是他们都需要通过自己的实现方式来执行,这就要得益于向上转型了。

          所以基于继承实现的多态可以总结如下:对于引用子类的父类类型,在处理该引用时,它适用于继承该父类的所有子类,子类对象的不同,对方法的实现也就不同,执行相同动作产生的行为也就不同。

          如果父类是抽象类,那么子类必须要实现父类中所有的抽象方法,这样该父类所有的子类一定存在统一的对外接口,但其内部的具体实现可以各异。这样我们就可以使用顶层类提供的统一接口来处理该层次的方法。

    基于接口实现的多态

          继承是通过重写父类的同一方法的几个不同子类来体现的,那么就可就是通过实现接口并覆盖接口中同一方法的几不同的类体现的。

          在接口的多态中,指向接口的引用必须是指定这实现了该接口的一个类的实例程序,在运行时,根据对象引用的实际类型来执行对应的方法。

          继承都是单继承,只能为一组相关的类提供一致的服务接口。但是接口可以是多继承多实现,它能够利用一组相关或者不相关的接口进行组合与扩充,能够对外提供一致的服务接口。所以它相对于继承来说有更好的灵活性。

      如 Map map =new HashMap ; List list=new ArraryList 都是基于接口实现的多态。

    经典实例

    public class A {
        public String show(D obj) {
            return ("A and D");
        }
     
        public String show(A obj) {
            return ("A and A");
        } 
     
    }
     
    public class B extends A{
        public String show(B obj){
            return ("B and B");
        }
        
        public String show(A obj){
            return ("B and A");
        } 
    }
     
    public class C extends B{
     
    }
     
    public class D extends B{
     
    }
     
    public class Test {
        public static void main(String[] args) {
            A a1 = new A();
            A a2 = new B();
            B b = new B();
            C c = new C();
            D d = new D();
            
            System.out.println("1--" + a1.show(b));
            System.out.println("2--" + a1.show(c));
            System.out.println("3--" + a1.show(d));
         //a2在编译期能确定的类型就是A,然后调用show(A),因为重载时只认静态类型,发现实际类型是B,并且B中重写了show(A a)方法,因此调用B.show(A a),结果为b and a System.out.println(
    "4--" + a2.show(b));
         //同上 System.out.println(
    "5--" + a2.show(c));
         //b中并没有重写show(D d)方法,因此直接调A.show(D d) System.out.println(
    "6--" + a2.show(d)); System.out.println("7--" + b.show(b)); System.out.println("8--" + b.show(c));
         //B继承了A中的show(D d)方法,因此结果为a and d System.out.println(
    "9--" + b.show(d)); } }

    运行结果:

    1--A and A
    2--A and A
    3--A and D
    4--B and A
    5--B and A
    6--A and D
    7--B and B
    8--B and B
    9--A and D

    经典实例分析

    上面程序中我们可以看出A、B、C、D存在如下关系

    我们把静态多态性看做是不同的方法,类A有两种方法,表示为A:show(D);A:show(A);

    类B继承于A,所以类B继承了类A的两种方法,表示为A:show(D);A:show(A);同时,类B有自己的方法,表示为B:show(B);B:show(A);这里B中重写了类A的show(A)方法,因此,B中的方法最终为A:show(D);B:show(B);B:show(A);

    类C,类D继承于B,所以类C类D的方法为:A:show(D);B:show(B);B:show(A);

    接下来,我们分析变量:

      A a1 = new A(); a1拥有类A的两种方法。A:show(D);A:show(A);

      A a2 = new B(); a2指向实例对象B,因此a2有类B的三种方法,又因为类B的向上转换,导致只保留类B中与类A中相同的方法,因此a2中的方法为A:show(D);B:show(A);

      B b = new B(); b拥有类B的三种方法A:show(D);B:show(B);B:show(A);

      C c = new C(); c拥有类C的三种方法A:show(D);B:show(B);B:show(A);

      D d = new D(); d拥有类D的三种方法A:show(D);B:show(B);B:show(A);

    接着,我们分析方法:

      a1.show(b):b是类型B的引用,a1不含有参数为类型B的方法,b可以向上转型为A,调用A:show(A);

      a1.show(c)同上;

      a1.show(d):d是类型D的引用,a1中有方法show(D),可以直接调用A:show(D)

      a2.show(b):b是类型B的引用,a2中不含有参数为类型B的方,b可以向上转型为A,调用B:show(A);

      a2.show(c)同上

      a2.show(d):a2中有方法show(D),可以直接调用A:show(d)

      b.show(b):b中有方法show(B),可以直接调用B:show(B);

      b.show(c):c是类型为C的引用,b中没有方法show(C),但是c可以向上转型为B,调用show(B);转型的时候,先转为直接父类,若没有找到直接父类对应的参数,再继续向上转;

      b.show(d):b中有方法show(D),可以直接调用A:show(D);

    动态绑定与静态绑定

    从编译和运行的角度看,重载,是指允许存在多个同名方法,而这些方法的参数不同。编译器根据方法不同的参数表,对同名方法的名称做修饰。对于编译器而言,这些同名方法就成了不同的方法。它们的调用地址在编译期就绑定了。Java的重载是可以包括父类和子类的,即子类可以重载父类的同名不同参数的方法。 所以:对于重载而言,在方法调用之前,编译器就已经确定了所要调用的方法, 这称为“早绑定”或“静态绑定”; 而对于多态,只有等到方法调用的那一刻,解释运行器才会确定所要调用的具体 方法,这称为“晚绑定”或“动态绑定”。重载在编译期间,编译器就可以进行区分,而重写,编译期间编译器无法知道调用的到底是哪个对象的方法,有可能是父类对象的,也有可能是子类对象的,因此只能运行时才能确定。

  • 相关阅读:
    《架构漫谈》有感
    《掌握需求过程》阅读笔记三
    《掌握需求过程》阅读笔记二
    《掌握需求过程》阅读笔记一
    《代码阅读方法与实现》阅读笔记三
    《代码阅读方法与实现》阅读笔记二
    《代码阅读方法与实现》阅读笔记一
    《软件需求模式》阅读笔记三
    《软件需求模式》阅读笔记二
    第二阶段个人总结06
  • 原文地址:https://www.cnblogs.com/alimayun/p/12910739.html
Copyright © 2020-2023  润新知