Java 多态——与C++的比较

时间:2021-03-28 16:08:26

学习了Java和C++之后,由于长期不使用C++,而java的基础知识掌握不牢,现在已经搞不清java多态了。现在先来谈谈java多态,稍后有时间再更新C++的多态,并进行比较~

一. Java的多态

首先什么是Java的多态?

多态是同一个行为具有多个不同表现形式或形态的能力。多态就是同一个接口,使用不同的实例而执行不同操作。

  • 实现多态的技术称为:动态绑定(dynamic binding),是指在执行期间判断所引用对象的实际类型,根据其实际的类型调用其相应的方法。
  • 多态的作用:消除类型之间的耦合关系。

Java中多态的实现方式:接口实现,继承父类进行方法重写,同一个类中进行方法重载。

1. 方法的重载

方法的重载是指在一个类中,出现多个方法名相同,但参数个数或参数类型不同的方法,则称为方法的重载。Java在执行具有重载关系的方法时,将根据调用参数的个数和类型区分具体执行的是哪个方法。

【例1】 定义一个名称为Calculate的类,在该类中定义两个名称为getArea()的方法(参数个数不同)和两个名称为draw()的方法(参数类型不同)。

 package test;

 public class Calculate {
     final float PI = 3.14159f;
     public float getArea(float r){
         float area = PI * r *r;
         return area;
     }
     public float getArea(float l,float w){
         float area = 1 * w;
         return area;
     }
     public void draw(int num){
         System.out.println("画"+num+"个任意形状的图形");

     }
     public void draw(String  shape){
         System.out.println("画一个"+shape);

     }
      public static void main(String[] args){
          Calculate calculate = new Calculate();
          float l = 20;
          float w = 30;
          float areaRectangle = calculate.getArea(l,w);
          System.out.println("求长为"+l+"宽为"+w+"的矩形的面积是:"+areaRectangle);
          float r = 7;
          float areaCirc = calculate.getArea(r);
          System.out.println("求半径为"+r+"的圆的面积是:"+areaCirc);
          int num = 7;
          calculate.draw(num);
          calculate.draw("三角形");

      }
 }

执行结果如下图所示:

Java 多态——与C++的比较

重载的方法之间并不一定必须有联系,但是为了提高程序的可读性,一般只重载功能相似的方法。

注意:在方法重载时,方法的返回值类型不能作为区分方法重载的标志。

2.方法的覆盖(重写)

当子类继承父类中所有可能被子类访问的成员方法时,如果子类的方法名与父类的方法名相同,那么子类就不能继承父类的方法,此时,称子类的方法覆盖了父类的方法。覆盖体现了子类补充或者改变父类方法的能力,通过覆盖,可以使一个方法在不同子类中表现出不同的行为。

【例2】定义动物类Animal及它们的子类,然后在Zoo类中分别创建各个子类对象,并调用子类覆盖父类的cry()方法。

(1)创建一个名称为Animal的类,在该类中声明一个成员方法cry():

 package example_2;

 public class Animal {

     public Animal() {

     }
     public void cry(){
         System.out.println("动物发出叫声!");
     }

 }

(2)创建一个Animal类的子类Dog类,在该类中覆盖父类的成员方法cry():

 package example_2;

 public class Dog extends Animal {

     public Dog() {
     }

     public void cry(){
         System.out.println("狗发出“汪汪.....”声!");
     }

 }

(3)再创建一个Animal类的子类Cat类,在该类中覆盖了父类的成员方法cry():

 package example_2;

 public class Cat extends Animal{

     public Cat() {
     }
     public void cry(){
         System.out.println("猫发出“喵喵....”声!");
     }

 }

(4)再创建一个Animal类的子类Cattle类,在该类中不定义任何方法:

 package example_2;

 public class Cattle extends Animal {

 }

(5)创建Zoo类,在该类的main()方法中分别创建子类Dog,Cat和Cattle的对象并调用它们的cry()成员方法:

 package example_2;

 public class Zoo {

     /**
      * @param args
      */
     public static void main(String[] args) {
         // TODO Auto-generated method stub
         Dog dog = new Dog();
         System.out.println("执行dog.cry();语句时的输出结果:");
         dog.cry();
         Cat cat = new Cat();
         System.out.println("执行cat.cry();语句时的输出结果:");
         cat.cry();
         Cattle cattle = new Cattle();
         System.out.println("执行cattle.cry();语句时的输出结果:");
         cattle.cry();

     }

 }

运行结果如下图所示:

Java 多态——与C++的比较

从上面的运行结果中可以看出,由于Dog类和Cat类都重写了父类的方法cry(),所以执行的是子类中的cry()方法,但是Cattle类没有重写父类的方法,所以执行的是父类中的cry()方法。事实上,在Zoo类中,如下写法更能体现多态性:

 package example_2;

 public class Zoo {

     /**
      * @param args
      */
     public static void main(String[] args) {
         // TODO Auto-generated method stub
         Animal animal;
         animal = new Dog();
         System.out.println("执行animal.cry();语句时的输出结果:");
         animal.cry();
         animal = new Cat();
         System.out.println("执行animal.cry();语句时的输出结果:");
         animal.cry();
         animal = new Cattle();
         System.out.println("执行animal.cry();语句时的输出结果:");
         animal.cry();

     }

 }

在进行方法覆盖时,需要注意一下几点:

  • 子类不能覆盖父类中声明为final或者static的方法。
  • 子类必须覆盖父类中声明为abstract的方法,或者子类也将该方法声明为abstract。
  • 子类覆盖父类中的同名方法时,子类中方法的声明也必须和父类中被覆盖的方法的声明一样。

3.接口实现