- 内容提要
- 前言
- 第 1 章 预备知识
- 第 2 章 开始学习 C++
- 第 3 章 处理数据
- 第 4 章 复合类型
- 第 5 章 循环和关系表达式
- 第 6 章 分支语句和逻辑运算符
- 第 7 章 函数——C++的编程模块
- 第 8 章 函数探幽
- 第 9 章 内存模型和名称空间
- 第 10 章 对象和类
- 第 11 章 使用类
- 第 12 章 类和动态内存分配
- 第 13 章 类继承
- 第 14 章 C++中的代码重用
- 第 15 章 友元、异常和其他
- 第 16 章 string 类和标准模板库
- 第 17 章 输入、输出和文件
- 第 18 章 探讨 C++新标准
- 附录 A 计数系统
- 附录 B C++保留字
- 附录 C ASCII 字符集
- 附录 D 运算符优先级
- 附录 E 其他运算符
- 附录 F 模板类 string
- 附录 G 标准模板库方法和函数
- 附录 H 精选读物和网上资源
- 附录 I 转换为 ISO 标准 C++
- 附录 J 复习题答案
13.2 继承:is-a 关系
派生类和基类之间的特殊关系是基于 C++继承的底层模型的。实际上,C++有 3 种继承方式:公有继承、保护继承和私有继承。公有继承是最常用的方式,它建立一种 is-a 关系,即派生类对象也是一个基类对象,可以对基类对象执行的任何操作,也可以对派生类对象执行。例如,假设有一个 Fruit 类,可以保存水果的重量和热量。因为香蕉是一种特殊的水果,所以可以从 Fruit 类派生出 Banana 类。新类将继承原始类的所有数据成员,因此,Banana 对象将包含表示香蕉重量和热量的成员。新的 Banana 类还添加了专门用于香蕉的成员,这些成员通常不用于水果,例如 Banana Institute Peel Index(香蕉机构果皮索引)。因为派生类可以添加特性,所以,将这种关系称为 is-a-kind-of(是一种)关系可能更准确,但是通常使用术语 is-a。
为阐明 is-a 关系,来看一些与该模型不符的例子。公有继承不建立 has-a 关系。例如,午餐可能包括水果,但通常午餐并不是水果。所以,不能通过从 Fruit 类派生出 Lunch 类来在午餐中添加水果。在午餐中加入水果的正确方法是将其作为一种 has-a 关系:午餐有水果。正如将在第 14 章介绍的,最容易的建模方式是,将 Fruit 对象作为 Lunch 类的数据成员(参见图 13.3)。
图 13.3 is-a 关系和 has-a 关系
公有继承不能建立 is-like-a 关系,也就是说,它不采用明喻。人们通常说律师就像鲨鱼,但律师并不是鲨鱼。例如,鲨鱼可以在水下生活。所以,不应从 Shark 类派生出 Lawyer 类。继承可以在基类的基础上添加属性,但不能删除基类的属性。在有些情况下,可以设计一个包含共有特征的类,然后以 is-a 或 has-a 关系,在这个类的基础上定义相关的类。
公有继承不建立 is-implemented-as-a(作为……来实现)关系。例如,可以使用数组来实现栈,但从 Array 类派生出 Stack 类是不合适的,因为栈不是数组。例如,数组索引不是栈的属性。另外,可以以其他方式实现栈,如链表。正确的方法是,通过让栈包含一个私有 Array 对象成员来隐藏数组实现。
公有继承不建立 uses-a 关系。例如,计算机可以使用激光打印机,但从 Computer 类派生出 Printer 类(或反过来)是没有意义的。然而,可以使用友元函数或类来处理 Printer 对象和 Computer 对象之间的通信。
在 C++中,完全可以使用公有继承来建立 has-a、is-implemented-as-a 或 uses-a 关系;然而,这样做通常会导致编程方面的问题。因此,还是坚持使用 is-a 关系吧。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论