C++学习笔记----4、用C++进行程序设计(五)---- 非复合与继承关系

2024-08-25 04:36

本文主要是介绍C++学习笔记----4、用C++进行程序设计(五)---- 非复合与继承关系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        在考虑类之间到底是什么关系时,要首先考虑一下是否真的有关系。不要搞有罪推定,直接问犯了什么罪,要先看是否是犯罪行为。不要让你对面向对象的设计的热情转变为大量不需要的类或者继承的类的关系中。

        有一个大家都容易陷入的怪圈就是在现实世界中很明显是相关的,但是用代码来实现却没有任凭实质性的关系。面向对象的层次关系需要模型化为函数关系,而不是人工关系。下图所示即为从形而上学或者层次上的关系是有意义的,但是从代码角度并没有什么意义。

        要避免这种无谓的继承的最好的方法就是首先画出设计的草图。对于每一个类,每一个继承类,写下你要为这些类赋予的属性与成员函数。如果你发现一个类没有自己的特定属性或成员函数就应该重新考虑你的设计了,或者是说所有的这些属性与成员函数都要在继承类中被彻底改写,当然了,要除掉那些前面提到的抽象基础类。

1、层次关系

        就像类A是类B的基础类,类B也可以是类C的基础类。面向对象的层次关系可以像这样模型化为多层关系。一个拥有多种动物的动物园的模拟可以被设计成每种动物都做为通用Animal类的派生类,如图所示:

        在对这些派生类进行编码时,你可能会发现许多代码是相似的。当这种情况发生时,你就要考虑将其放到一个通用的父类中了。意识到Lion与Panther移动方式一样并且吃同样的饲料,就需要来一个BigCat类。同样可以将Animal分成包括WaterAnimal与Marsupial。下图所示即为利用了这种共同性的一种层次关系设计。

        生物学家看到这样的层次关系一定会很失望----penguin与dolphin真的不是一个动物家族。然而,它要表达的是一个在代码上的观点,你需要平衡真实世界与共享功能之间的关系,不要搞了一个什么都对,但是不解决任何问题的方案,这样的方案没有意义,还是那个观点--不管黑猫白猫,抓住老鼠就是好猫。即使两个东西在现实世界中非常有关系,但在代码中没有任何关系,因为他们不共享任何功能。你可以将动物很容易地划分为哺乳动物与鱼类,但这没有提炼出任何基类的共同性,有什么意义呢?

        另外一个重要的观点就是要有组织层次关系的方法。前面的设计大部分都是以动物如何移动进行组织的。如果要以动物的饮食或高度进行组织呢?其层次结构就会有很大不同。重要的是,类要怎么用。用途决定了类层次关系的设计。

        好的面向对象的层次关系要完成如下任务:

  • 将类组织成有意义的功能关系。
  • 支持代码复用,将通用功能提炼至基类。
  • 避免在派生类中对父类的功能进行覆盖,除非父类为抽象基础类。

2、多重继承

        到目前为止的所有例子都是单个的继承链条。换句话说,一个给定的类,最多只有一个直接的父类。不一定非要这样。通过多重继承,一个类可以有多于一个的基类。

        下图展示了一个多重继承的设计。仍然有一个叫做Animal的基类,它是通过大小来分的,还有一个单独的层次结构分类为通过饮食,第三个是关于移动方式。每种动物这样的话就是这三个类的派生类,在下面图中不同行中展示。

        在用户操作界面中,如果可以单击一个图片,那这个类看起来就是按钮与图片类,这样其实现可能就会继承自Image与Button两个类,如图所示:

        在有些情况下多重继承是很有用的,但要记住也有许多不好的方面。万事皆有其规律,其实越是简单的,越是好的,越是高效的,复杂的设计只是为了解决特定的问题,如果你把所有问题都复杂化,以此来显示你的高能,从根本上来说, 你的想法与做法其实都错了。其实许多程序员都不喜欢多重继承,C++是支持这种关系的,Java除了支持从多个接口(抽象基础类)进行派生,其他的多重继承都被Java抛弃掉了。对于多重继承的批评观点有几个原因。

        首先,多重继承的观感很复杂。你从上面的那个动物的图中就可以看出,当有多个层次关系和交叉线时,即使一个简单的类图也可以变得很复杂。类的层次结构被设计用来帮助程序员理解代码之间的关系。用了多重继承,一个类就有了多个相互之间无关的父类。这么多的类给你的对象贡献代码,你真的能够跟踪知道到底发生了什么吗?

        其次,多重继承会在破坏清晰的层次结构。在那个动物的例子中,用多重继承的方法意味着Animal的基类的意义就不大了,因为描述动物的代码现在分成了三个单独的层次结构。在上面相应的图中其设计展示了有清晰的三类层次结构,不难想像,它们是怎么搅和在一起的。例如,你突然意识到所有的Jumpers不但移动方式相同,还吃同样的东西?因为有单独层次结构,在不增加另外一个派生类的情况下,没有办法将移动方式与饮食的概念放在一起。

        第三,多重继承的实现非常复杂。如果两个基类用不同的方法实现了同样的成员函数?你能让这两个基类派生于一个通用的基础类吗?这种可能性使实现复杂化,因为在代码中结构化这样细致的关系不管是对于读者还是作者都太困难了。

        其他语言不用多重继承是因为通常这种方式是可以避免的。重新思考一下你的层次关系,如果你能对项目的设计进行控制,就可以避免多重继承的发生。

3、混合类

        混合类是类间关系的另一种类型。在C++中,实现混合类的一种方式是在语法上非常像多重继承,但是语义上非常不同。混合类回答的是这样的问题,“这个类另外还能做什么?”,答案通常是,“它是。。。的”。混合类就是一种不全是继承关系的可以给类增加功能的方式。你可以把它认为是一种共享什么的关系。

        再回到动物园的那个例子,你可能想给一些动物加上“宠物的”标签。也就是说,有些动物游客参观动物园时可以认为是宠物,它不会咬人也不会挠人。你可能会将所有宠物动物支持“成为宠物”的行为。因为宠物动物并不具有其他共同的另外的东西,不需要破坏既有的设计好的层次结构,“宠物的”就是一个很好的混合类。

        混合类常用于用户接口。除了说PictureButton类既是Image又是Button外,还可以说Image是Clickable。一个电脑桌面上的文件夹图标就是一个可以DraggablegnClickable的Image。软件开发者可以造出许多有趣的对象。

        混合类与基类的不同与你考虑类与代码的不同有关。通常来说,混合类比多重继承容易消化,因为其限制在一定范围内。Pettable混合类只是给已有类增加了一种行为。Clickable混合类可以只是增加"mouse down"与“mouse up”的行为。还有,混合类很少有大的层次结构,所以就不会对功能性进行交叉感染。

这篇关于C++学习笔记----4、用C++进行程序设计(五)---- 非复合与继承关系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1104560

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名