美文网首页
JavaSE基础知识学习-----多态

JavaSE基础知识学习-----多态

作者: Waldeinsamkeit4 | 来源:发表于2018-02-25 12:39 被阅读0次

    多态

    Java面向对象编程有三大特性:封装,继承和多态
    封装隐藏类的内部具体实现机制,保护数据,对外界隐藏内部细节,只向外部提供它所允许访问的方法
    继承是为了复用代码和实现向上转型,当然继承也是为多态做准备。
    多态可以说是Java面向对象的精华所在。

    什么是多态

    多态是指允许不同子类型的对象对同一消息做出不同的响应,简单来讲,就是用同样的对象调用同样的方法但是却做了不同的事情。也可以理解为一个事物的多种表现形态。
    多态性分为编译时的多态性和运行时的多态性。例如:
    方法重载:实现的是编译时的多态性
    方法重写:实现的是运行是的多态性
    子类对象的多态性,这个是Java多态性中最常用的事。

    实现多态的条件

    1.方法重写,也就是说要有继承吧,只有有了继承,子类重写了父类已有的或抽象的方法
    2.对象造型,也就是说要有父类的引用指向子类对象
    只有这样,同样的引用调用同样的方法就会作出不同的响应,简单来讲如下代码所示:
    父类Person有如下方法:

    public void walk(){
        System.out.println("人走路");
    }
    
    public void eat(){
        System.out.println("人吃饭");
    }
    

    如果子类Man类没有重写这两个方法,那么我们做如下的事情:

        Person man = new Man();
        man.walk();
    

    输出结果:

    人走路
    

    综上所述:父类引用man指向的是子类对象,这个就是子类对象的多态性(也叫向上转型),man可以调用父类的方法,前提是子类没有重写父类方法。
    如果子类Man类重写了父类方法

    public void walk(){
        System.out.println("男人应该挺拔的走路");
    }
    
    public void eat(){
        System.out.println("男人应该多吃肉");
    }
    

    同样的调用:

    Person man = new Man();
    man.walk();
    

    结果就是:

    男人应该挺拔的走路
    

    综上所述:子类对象重写了父类方法,那么man调用父类方法,执行的就是子类的方法。这个也虚拟方法调用,说到这里你应该可以猜到,什么是多态了吧,就是同样的父类对象应用,指向不同的子类对象,就会作出不同的响应,例如子类Woman类也重写了父类的方法:

    public void walk(){
        System.out.println("女人应该温柔的走路");
    }
    
    public void eat(){
        System.out.println("女人应该少吃肉多吃水果蔬菜");
    }
    

    调用方法:

    Person man = new Man();
    Person woman = new Woman();
    man.walk();
    woman.walk();
    

    结果为:

    男人应该挺拔的走路
    女人应该温柔的走路
    

    综上所述:应该是很清晰了什么是子类对象的多态性,就是你相同的父类引用,不同的子类对象,就会响应对应的子类对象的方法。

    方法重载和方法重写

    方法重载和方法重写两者都是实现多态的方式,区别在于方法重载实现的是编译时的多态,方法重写实现的是运行时的多态,重载发生在同一个类中,同名的方法有不同的参数列表。

    注意:方法的重载与返回值无关,简称两同一不同,同一个类,同一个方法,不同的参数列表

    方法的重写发生在子类和父类之间,重写要求子类方法返回值,方法名,参数列表与父类爆出一致,权限修饰符大于父类,同时要同为static或同为非static。

    注意:构造器不能被继承,所以不能被重写,但是却可以被重载

    多态的使用例子

    例如在父类Person类中有如下方法:

    public void show(Person person){
        System.out.println("这是父类show方法");
    }
    

    子类Man类重写了子类方法:

    public void show(Person person){
        System.out.println("这是子类show方法");
    }
    

    如果实例化Man对象,调用man的show(Person person)方法,就应该传入Person对象,这个时候就可以传入父类对象的引用。

    Person man = new Man();
    man.show(man);
    

    父类引用指向不同的子类对象,作出不同的响应。

    多态经典案例

    摘自http://blog.csdn.net/thinkGhoster/archive/2008/04/19/2307001.aspx

        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));  
            System.out.println("4--" + a2.show(b));  
            System.out.println("5--" + a2.show(c));  
            System.out.println("6--" + a2.show(d));  
            System.out.println("7--" + b.show(b));  
            System.out.println("8--" + b.show(c));  
            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  
    

    结果分析如下:

    System.out.println("1--" + a1.show(b)); //a1是A的引用指向是A对象,故调用A中的show方法,传入的是b,b指向的是B对象,因为B继承与A,故传入的也可认为是A,结果为A and A
    System.out.println("2--" + a1.show(c)); //c指向C对象,C继承B,B继承A,相当于传入A,结果为A and A
    System.out.println("3--" + a1.show(d)); //d指向D对象,在A中有传入D对象的方法,结果为 A and D
    System.out.println("4--" + a2.show(b)); //a2是父类引用,指向子类对象,按理说调用show方法会执行子类B的show方法,传入的是b,b指向的是B对象,
                                            //但是B中的show(B)是子类特有的方法,故不能被调用,传入的是B,B继承与A,相当于传入A,B类重写了show(B)方法,结果为B and A
    System.out.println("5--" + a2.show(c)); //传入c,执行B中show(C)的方法,没有,C继承B,执行B中show(B),但是这是B中特有方法,不能被调用,
                                            //B继承A,调用B中show(A),结果B and A
    System.out.println("6--" + a2.show(d)); //传入D,执行B中show(D)方法,没有,但是A中有就会执行父类show(D)方法,所以结果为:A and D
    System.out.println("7--" + b.show(b)); //b指向B,传入B, 结果为B and B
    System.out.println("8--" + b.show(c)); //传入c,执行B中show(C),没有,C继承B,执行B中show(B),结果为:B and B
    System.out.println("9--" + b.show(d)); //传入d,执行B中show(D)方法,没有,当时B继承A,是存在A中show(D)的方法,结果为 A and D 
    

    案例总结

    对于a2.show(c)来讲,a2是A的应用,故先去A中找show(C)方法(this.show(O)),A中没有,按理应该去A的父类中找(super.show(O)),但是A没有父类,故只能在A中找show(c的父类)方法(this.show(super)),c的父类有B和A,A中找到show(A)的方法,但是B重写了该方法,故执行子类的方法,所以结果为B and A。找的顺序是this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。

    相关文章

      网友评论

          本文标题:JavaSE基础知识学习-----多态

          本文链接:https://www.haomeiwen.com/subject/tbcrxftx.html