`

java 继承、多态、重载和重写

    博客分类:
  • Java
阅读更多

什么是多态?它的实现机制是什么呢?重载和重写的区别在那里?这就是这一次我们要回顾的四个十分重要的概念:继承、多态、重载和重写。

继承(inheritance)

简单的说,继承就是在一个现有类型的基础上,通过增加新的方法或者重定义已有方法(下面会讲到,这种方式叫重写)的方式,产生一个新的类型。继承是面向对象的三个基本特征--封装、继承、多态的其中之一,我们在使用JAVA时编写的每一个类都是在继承,因为在JAVA语言中,java.lang.Object类是所有类最根本的基类(或者叫父类、超类),如果我们新定义的一个类没有明确地指定继承自哪个基类,那么JAVA就会默认为它是继承自Object类的。

我们可以把JAVA中的类分为以下三种:

  1. 类:使用class定义且不含有抽象方法的类。
  2. 抽象类:使用abstract class定义的类,它可以含有,也可以不含有抽象方法。
  3. 接口:使用interface定义的类。


在这三种类型之间存在下面的继承规律:

  • 类可以继承(extends)类,可以继承(extends)抽象类,可以继承(implements)接口。
  • 抽象类可以继承(extends)类,可以继承(extends)抽象类,可以继承(implements)接口。
  • 接口只能继承(extends)接口。


请注意上面三条规律中每种继承情况下使用的不同的关键字extends和implements,它们是不可以随意替换的。大家知道,一个普通类继承一个接口后,必须实现这个接口中定义的所有方法,否则就只能被定义为抽象类。我在这里之所以没有对implements关键字使用“实现”这种说法是因为从概念上来说它也是表示一种继承关系,而且对于抽象类implements接口的情况下,它并不是一定要实现这个接口定义的任何方法,因此使用继承的说法更为合理一些。

以上三条规律同时遵守下面这些约束:

  1. 类和抽象类都只能最多继承一个类,或者最多继承一个抽象类,并且这两种情况是互斥的,也就是说它们要么继承一个类,要么继承一个抽象类。
  2. 类、抽象类和接口在继承接口时,不受数量的约束,理论上可以继承无限多个接口。当然,对于类来说,它必须实现它所继承的所有接口中定义的全部方法。
  3. 抽象类继承抽象类,或者实现接口时,可以部分、全部或者完全不实现父类抽象类的抽象(abstract)方法,或者父类接口中定义的接口。
  4. 类继承抽象类,或者实现接口时,必须全部实现父类抽象类的全部抽象(abstract)方法,或者父类接口中定义的全部接口。


继承给我们的编程带来的好处就是对原有类的复用(重用)。就像模块的复用一样,类的复用可以提高我们的开发效率,实际上,模块的复用是大量类的复用叠加后的效果。除了继承之外,我们还可以使用组合的方式来复用类。所谓组合就是把原有类定义为新类的一个属性,通过在新类中调用原有类的方法来实现复用。如果新定义的类型与原有类型之间不存在被包含的关系,也就是说,从抽象概念上来讲,新定义类型所代表的事物并不是原有类型所代表事物的一种,比如黄种人是人类的一种,它们之间存在包含与被包含的关系,那么这时组合就是实现复用更好的选择。下面这个例子就是组合方式的一个简单示例:

Java代码 复制代码
  1. public class Sub {   
  2.     private Parent p = new Parent();   
  3.   
  4.     public void doSomething() {   
  5.         // 复用Parent类的方法   
  6.         p.method();   
  7.         // other code   
  8.     }   
  9. }   
  10.   
  11. class Parent {   
  12.     public void method() {   
  13.         // do something here   
  14.     }   
  15. }  
public class Sub {
	private Parent p = new Parent();

	public void doSomething() {
		// 复用Parent类的方法
		p.method();
		// other code
	}
}

class Parent {
	public void method() {
		// do something here
	}
}


当然,为了使代码更加有效,我们也可以在需要使用到原有类型(比如Parent p)时,才对它进行初始化。

使用继承和组合复用原有的类,都是一种增量式的开发模式,这种方式带来的好处是不需要修改原有的代码,因此不会给原有代码带来新的BUG,也不用因为对原有代码的修改而重新进行测试,这对我们的开发显然是有益的。因此,如果我们是在维护或者改造一个原有的系统或模块,尤其是对它们的了解不是很透彻的时候,就可以选择增量开发的模式,这不仅可以大大提高我们的开发效率,也可以规避由于对原有代码的修改而带来的风险。

多态(Polymorphism)

多态是又一个重要的基本概念,上面说到了,它是面向对象的三个基本特征之一。究竟什么是多态呢?我们先看看下面的例子,来帮助理解:

Java代码 复制代码
  1. //汽车接口   
  2. interface Car {   
  3.     // 汽车名称   
  4.     String getName();   
  5.   
  6.     // 获得汽车售价   
  7.     int getPrice();   
  8. }   
  9.   
  10. // 宝马   
  11. class BMW implements Car {   
  12.     public String getName() {   
  13.         return "BMW";   
  14.     }   
  15.   
  16.     public int getPrice() {   
  17.         return 300000;   
  18.     }   
  19. }   
  20.   
  21. // 奇瑞QQ   
  22. class CheryQQ implements Car {   
  23.     public String getName() {   
  24.         return "CheryQQ";   
  25.     }   
  26.   
  27.     public int getPrice() {   
  28.         return 20000;   
  29.     }   
  30. }   
  31.   
  32. // 汽车出售店   
  33. public class CarShop {   
  34.     // 售车收入   
  35.     private int money = 0;   
  36.   
  37.     // 卖出一部车   
  38.     public void sellCar(Car car) {   
  39.         System.out.println("车型:" + car.getName() + "  单价:" + car.getPrice());   
  40.         // 增加卖出车售价的收入   
  41.         money += car.getPrice();   
  42.     }   
  43.   
  44.     // 售车总收入   
  45.     public int getMoney() {   
  46.         return money;   
  47.     }   
  48.   
  49.     public static void main(String[] args) {   
  50.         CarShop aShop = new CarShop();   
  51.         // 卖出一辆宝马   
  52.         aShop.sellCar(new BMW());   
  53.         // 卖出一辆奇瑞QQ   
  54.         aShop.sellCar(new CheryQQ());   
  55.         System.out.println("总收入:" + aShop.getMoney());   
  56.     }   
  57. }  
//汽车接口
interface Car {
	// 汽车名称
	String getName();

	// 获得汽车售价
	int getPrice();
}

// 宝马
class BMW implements Car {
	public String getName() {
		return "BMW";
	}

	public int getPrice() {
		return 300000;
	}
}

// 奇瑞QQ
class CheryQQ implements Car {
	public String getName() {
		return "CheryQQ";
	}

	public int getPrice() {
		return 20000;
	}
}

// 汽车出售店
public class CarShop {
	// 售车收入
	private int money = 0;

	// 卖出一部车
	public void sellCar(Car car) {
		System.out.println("车型:" + car.getName() + "  单价:" + car.getPrice());
		// 增加卖出车售价的收入
		money += car.getPrice();
	}

	// 售车总收入
	public int getMoney() {
		return money;
	}

	public static void main(String[] args) {
		CarShop aShop = new CarShop();
		// 卖出一辆宝马
		aShop.sellCar(new BMW());
		// 卖出一辆奇瑞QQ
		aShop.sellCar(new CheryQQ());
		System.out.println("总收入:" + aShop.getMoney());
	}
}


运行结果:

  1. 车型:BMW  单价:300000
  2. 车型:CheryQQ  单价:20000
  3. 总收入:320000


继承是多态得以实现的基础。从字面上理解,多态就是一种类型(都是Car类型)表现出多种状态(宝马汽车的名称是BMW,售价是300000;奇瑞汽车的名称是CheryQQ,售价是2000)。将一个方法调用同这个方法所属的主体(也就是对象或类)关联起来叫做绑定,分前期绑定和后期绑定两种。下面解释一下它们的定义:

  1. 前期绑定:在程序运行之前进行绑定,由编译器和连接程序实现,又叫做静态绑定。比如static方法和final方法,注意,这里也包括private方法,因为它是隐式final的。
  2. 后期绑定:在运行时根据对象的类型进行绑定,由方法调用机制实现,因此又叫做动态绑定,或者运行时绑定。除了前期绑定外的所有方法都属于后期绑定。


多态就是在后期绑定这种机制上实现的。多态给我们带来的好处是消除了类之间的耦合关系,使程序更容易扩展。比如在上例中,新增加一种类型汽车的销售,只需要让新定义的类继承Car类并实现它的所有方法,而无需对原有代码做任何修改,CarShop类的sellCar(Car car)方法就可以处理新的车型了。新增代码如下:

Java代码 复制代码
  1. // 桑塔纳汽车   
  2. class Santana implements Car {   
  3.     public String getName() {   
  4.         return "Santana";   
  5.     }   
  6.   
  7.     public int getPrice() {   
  8.         return 80000;   
  9.     }   
  10. }  
// 桑塔纳汽车
class Santana implements Car {
	public String getName() {
		return "Santana";
	}

	public int getPrice() {
		return 80000;
	}
}



重载(overloading)和重写(overriding)

重载和重写都是针对方法的概念,在弄清楚这两个概念之前,我们先来了解一下什么叫方法的型构(英文名是signature,有的译作“签名”,虽然它被使用的较为广泛,但是这个翻译不准确的)。型构就是指方法的组成结构,具体包括方法的名称和参数,涵盖参数的数量、类型以及出现的顺序,但是不包括方法的返回值类型,访问权限修饰符,以及abstract、static、final等修饰符。比如下面两个就是具有相同型构的方法:

Java代码 复制代码
  1. public void method(int i, String s) {   
  2.     // do something   
  3. }   
  4.   
  5. public String method(int i, String s) {   
  6.     // do something   
  7. }  
public void method(int i, String s) {
	// do something
}

public String method(int i, String s) {
	// do something
}


而这两个就是具有不同型构的方法:

Java代码 复制代码
  1. public void method(int i, String s) {   
  2.     // do something   
  3. }   
  4.   
  5. public void method(String s, int i) {   
  6.     // do something   
  7. }  
public void method(int i, String s) {
	// do something
}

public void method(String s, int i) {
	// do something
}


了解完型构的概念后我们再来看看重载和重写,请看它们的定义:

  • 重写,英文名是overriding,是指在继承情况下,子类中定义了与其基类中方法具有相同型构的新方法,就叫做子类把基类的方法重写了。这是实现多态必须的步骤。
  • 重载,英文名是overloading,是指在同一个类中定义了一个以上具有相同名称,但是型构不同的方法。在同一个类中,是不允许定义多于一个的具有相同型构的方法的。


我们来考虑一个有趣的问题:构造器可以被重载吗?答案当然是可以的,我们在实际的编程中也经常这么做。实际上构造器也是一个方法,构造器名就是方法名,构造器参数就是方法参数,而它的返回值就是新创建的类的实例。但是构造器却不可以被子类重写,因为子类无法定义与基类具有相同型构的构造器。

分享到:
评论

相关推荐

    继承与多态,重载和重写

    总结起来,继承、多态、重载和重写是面向对象编程中的核心概念,它们相互关联,共同构建出灵活、可扩展的代码结构。通过继承,我们可以构建类的层次结构,多态则提供了代码的通用性和灵活性。而重载和重写则是实现...

    java 重载,重写以及继承,多态的区别

    Java 中的重载、重写、继承和多态是四个不同的概念,它们之间存在一定的关系,但也存在明显的区别。 重载(Overload) 重载是指在同一个类中可以定义多个同名的方法,但这些方法的形参个数或类型必须不同。编译器...

    [Java]重载,重写以及继承,多态的区

    在Java编程语言中,重载、重写、继承与多态是四个非常核心的概念,它们在面向对象编程中扮演着至关重要的角色。接下来,我们将详细探讨这些概念,并通过实例加以说明。 ### 重载(Overload) 重载是指在同一个类中...

    java 接口 类 继承 多态的简单实例

    多态分为编译时多态(方法的重载,Overload)和运行时多态(方法的重写,Override)。 - **重载(Overload)**:是指在同一个类中可以有多个同名方法,但参数列表必须不同(参数个数、类型或顺序),这使得我们可以...

    java中的重载和重写的区别

    Java中的重载(Overloading)和重写(Overriding)是面向对象编程中两个重要的概念,它们体现了Java的多态性。多态性是面向对象编程的三大特性之一,另外两个是封装和继承。 首先,让我们详细解释一下方法的重载。重载...

    java中继承与多态的题目

    * 实现继承和多态的应用,例如方法重写和方法重载 实验结果 通过实验,我们可以看到继承和多态的应用,可以简化程序的设计和实现,提高代码的重用性和可维护性。 结论 本资源摘要信息涵盖了 Java 中继承和多态的...

    java基础继承封装多态

    在 Java 中,可以通过方法重写和方法重载来实现多态。方法重写是子类继承父类的方法,并提供不同的实现。方法重载是提供多个同名方法,但是参数类型不同。 面向对象的设计 面向对象的设计是 Java 中的一种设计模式...

    JAVA进阶-Java继承,多态,重写,修饰符以及super与this使用,Oject对象

    在Java中,多态有两种形式:编译时多态(方法重载)和运行时多态(方法重写)。运行时多态是通过接口或继承实现的,它依赖于对象的实际类型而非声明类型。 3. **重写(Override)**: 当子类需要覆盖或修改父类中...

    java 多态重载 PPT

    多态在Java中主要体现在方法的重载(Overloading)和覆盖(Overriding)上。 1. 多态(Polymorphism) 多态的概念指的是同一种行为可以有不同的表现形式。在Java中,多态主要通过继承和接口来实现。例如,`Person`...

    [Java]重载重写以及继承多态的区别.doc

    ### Java中的重载、重写、继承与多态 #### 一、重载(Overloading) 在Java中,**重载**是指在一个类内可以有多个名称相同但参数列表不同的方法。这些方法可以通过不同的参数类型、数量或者顺序来进行区分。需要...

    Java继承和多态详解及其综合应用

    接着讨论了多态的定义、方法重载和重写的区别、向上转型与向下转型的应用。最后,通过具体的设计模式示例展示了继承和多态在软件架构中的重要作用和实际应用,如工厂模式、策略模式等。文章还包括了一些进阶技巧,如...

    封装、继承、多态.docx

    多态可以分为两种形式:方法重载和方法重写。 1. 方法重载:是指多个方法具有相同的方法名,但不同的参数列表。 2. 方法重写:是指子类对父类的方法进行重写,以实现不同的行为。 LSP(里氏代换原则): 1. 子类...

    Java继承与多态的全面解析及其综合应用

    内容概要:本文档详尽介绍了Java继承和多态的基本概念及综合应用。涵盖Java继承的基础,如关键字extends与super的使用、构造函数继承及方法重写等;还探讨了Java多态的特点,如方法重载与重写、向上转型与向下转型的...

    java继承、多态的简单程序.zip

    这个压缩包"java继承、多态的简单程序.zip"显然包含了一些示例代码,旨在帮助初学者理解这两个概念。接下来,我们将深入探讨Java中的继承和多态性。 ### 继承(Inheritance) 在Java中,继承允许一个类(子类)...

    北大青鸟javaOOP多态Ch03作业全集.rar

    在Java中,多态主要通过接口和继承来实现。 1. 继承:子类继承父类,可以获得父类的所有非私有属性和方法。在Java中,一个类只能单继承一个父类,但可以实现多个接口,这就是多态的基础。继承的多态性体现在父类...

    重载,重写,覆盖,多态

    在面向对象编程语言中,如Java、C#等,重载(Overloading)、重写(Overriding)、覆盖(也称为重写)以及多态(Polymorphism)是几个非常重要的概念。它们之间既有关联又有区别,本文将详细解释这些概念,并探讨...

    java 重载,继承,重写和多态的区别. doc

    Java编程语言中的重载、继承、重写和多态是面向对象编程的四个核心概念,它们各自具有独特的功能和用途。 1. **重载(Overloading)**: 重载发生在同一个类内部,指的是可以有多个同名方法,但这些方法的参数列表...

    继承,重载,重写,多态,抽象类,接口,final,static

    继承,重载,重写,多态,抽象类,接口,final,static

    Java编程入门-深入解析继承和多态的基本原理与应用案例

    内容概要:本文详细介绍了Java编程中继承和多态的概念及其应用,涵盖从基础到进阶的内容,包括继承的基本概念、super关键字的应用、方法重载与重写、instanceof关键字的作用以及继承与多态的具体示例。 适合人群:...

Global site tag (gtag.js) - Google Analytics