(Boolan) C++面向对象高级编程(三)

2024-04-18 08:08

本文主要是介绍(Boolan) C++面向对象高级编程(三),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天来看一看关于面向对象的编程的一些内容,也就是关于类和类之间的关系。类和类之间的关系其实一种抽象的描述现实世界的方法。

现实生活中的事物不都和之前提到的String或者Complex一样与其他事务没有太多的联系,其实从不同角度看到事物往往会得出不同的结论。比如,生物界的划分经常是以界门纲目科属种这样的思路来划分的,找出其共性,然后再向上抽象 得出一个概述性的大类。比如,生物学上,人被分类为真核总界 动物界 后生动物亚界 后口动物总门 脊索动物门 脊椎动物亚门 羊膜总纲 哺乳纲 真兽亚纲 灵长目 类人猿亚目 狭鼻猴次目 类人猿超科 人科 人亚科 人族 人属 智人种;猫被分类为真核总界 动物界 后生动物亚界 后口动物总门 脊索动物门 脊椎动物亚门 羊膜总纲 哺乳纲 真兽亚纲 食肉目 猫亚科 猫属 猫种(引自百度百科)。对比猫和人不难发现,在真兽亚纲之前和人类没有区别,但是从目开始,人属于灵长目、而猫咪属于食肉目,从才分道扬镳。那这些分类对于解释这个世界到底有什么帮助吗?这个帮助是很自然的,比如每往下细分一个层次,就意味着区分的细节越细,而越往上则越来越抽象。又比如,真核生物中分四个界:原生生物界、真菌界、植物界和动物界。就拿最熟悉的植物界和动物界来看,他们的区别是显著的,但却别的点往往是粗糙的。

毕竟我对生物学没有太深的了解,但是这样一种分类方法在认识和解释世界的过程中却是通用的。我们希望通过计算机来人和世界结合,那么还是需要对这样的认识方法有一些了解。

C++相较于C的发展,主要是出现了对象的概念,而对象是由各式各样的class构造而成,那么class和class之间的关系就变得特别重要。因为人和人之间就像是对象和对象之间的关系,他们都是同一种class所构造而成的,但是属性却不相同(性别、名 字、名族、肤色、身高、体重、性格,生命周期都不一定相同),虽然存在着这些差异,但每个人都还是属于人类的 (class 人)。但是人和猫的却别不仅仅是对象之间的差别了,更是物种之间的差别了。但是这两个class之间也并不是完全没有相同点,比如这两个物种小的时候都得喝奶等等的相同点,把这喝奶的的这个特点向上抽取,那么就得到了一个新的class(哺乳纲)[ 可以是 抽象类 ],人和猫这两个class和新产生的class之间发生关系就行了,因为这足够表示出这俩的关系了。最后,我们会得到一棵像树一样庞大的一个体系,通过这个体系,我们就能很方便的解释节点(class之间的关系)。

类的基本思想是 数据抽象(data abstraction) 封装(encapsulation)。数据抽象就是一种依赖于 接口(Interface) 实现(implementation)。类的接口包括用户所能执行的操作;类的实现则包括类的数据成员、负责接口实现的函数体以及定义类所需的各种私有函数。

通过这样的抽象,不能方便描述class之间的关系,还更好的将其中的数据进行封装。再拿人和猫来举例,猫身上通常都一身漂亮的皮毛,但人身上基本没有(如果说人身上的汗毛其实不一定比猫少,那么这就是种诡辩了)。这样会有一个巨大的好处,让猫身上的毛很直接的就属于猫咪,而不会出现在人类的class中。这就实现了非常好的 封装 的效果。

这样的操作,对于数据的封装有着巨大的好处,但是,这之间也又带有了一个巨大的方便,比如,猫咪身上的毛,是猫咪自己的私有属性,这时候,同样猫也可以具有一个函数,这个函数专门用于操作他的私有成员属性(猫身上的毛),而人则不需要这样的函数(方法),原因是因为没有人类这样的私有成员属性。

那么为了描述这样的关系,我们主要讨论class之间的3中关系。

三种关系

  • Inheritance(继承)

    • 表示is-a的关系

      struct _List_node_hase
      {_List_node_base * _M_next;_List_node_base * _M_prev;
      };
      template <typename _Tp>
      struct _List_node: public _List_node_base
      {_Tp _M_data;
      };
    • UML图


    • 父类的数据在public的继承关系下,能够被完全继承

    • 父类函数的调用权也可以被子类进程
    • Inheritance关系下的构造和析构


      • 构造:由内而外
        • Derived的构造函数首先调用Base的default构造函数,然后才执行自己
      • 析构:由外而内
        • Derived的析构函数首先执行自己,然后才调用Base的析构函数
    • 虚函数

      • non-virtual函数:不希望Derived class重新定义(override)
      • virtual函数:希望Derived class 重新定义,且已有默认的定义
      • pure virtual函数:希望Derived class一定要重新定义(override),并且没有默认定义
  • Composition(复合)

    • has-a的关系
    • 我把他定义为人和长在身上的器官的关系

        template<class T, class Sequence = deque<T> >class queue {....protected :Sequence c; //底层容器public://以下完全利用c的操作函数完成bool empty() const { return c.empty(); }  //其中的操作均为调用 Sequence c;中所实现的功能size_type size() const { return c.size(); }....}
      • 在queue中,持有一个Sequence c 的对象,queue和Sequence之间的关系就是Composition
      • 其中UML图为

        queue 和 sequence 为Composition的关系
        • 其中黑色菱形一段表示持有另外一个class的对象
      • Adapter设计模式主要使用了Composition来实现。Adapter可以很好的将部分功能封装和开放
    • Composition所占用内存的大小等于他持有的所有的成员所占内存的总和

    • Composition关系下的构造和析构
      • 构造:由内而外(类似于打包裹)
        • Container的构造函数首先调用Component的default构造函数,然后才执行自己的构造函数
          Container:: Container(...) : Component()//Component()由编译器完成添加
          • 如果不满意调用默认构造函数那么需要自己来写需要的构造函数在初始化列表
      • 析构:由外而内(相当于拆包裹)
        • Container的析构函数首先执行,然后才会调用Component的析构函数
          Container:: ~Container(...){ ... ~Component(); } //~Component()有编译器添加
      • 只需要将自己的构造和析构完成即可,顺序会有编译器自动来完成
      • 从生命周期的角度来看,这两个对象之间的关系为“同生共死”
      • 我理解Composition类似于人身上的器官,这些器官由他自己的功能,比如我们需要吸收能力来维持生命,那么我们只需要调用我们自己的消化系统的消化功能(消化系统.消化(食物);),类似于这样的过程,我们就完成了吸收营养的过程,但实际这个过程不是人类这个class完成的。并且一旦人死去,这些器官也会死去(这里可不讨论捐献移植啊!!!),并且也是人先死,之后的器官一个一个死去。
  • Delegation(委托)

    • Composition by reference(不持有需要的对象,只持有目标对象的指针)
    • 我把这个关系定义为我们和东西关系

      class StringRep;
      class String{.....private:StringRep* rep;//持有目标对象的指针
      }
    • UML图示为


    • 用指针相连,所以两个对象的生命周期不同
  • 对于这个关系,我把他理解为我们和东西关系,比如我们和手机,我们需要和远处的朋友聊天,我们可以用手机来完成(只需要调用手机的打电话函数手机->打电话(电话号码);),就相当于我们用我们持有的手机的指来打了一个电话。那为什么是指针呢,因为我们并不是真正拥有一部手机,我们拥有的只是他的使用权和处置权(说白了就是他们都是我们的身外之物,生不带来死不带去,在过去,大家都共用一部电话,那时候只有使用权,也就反应了指针所指对象的特点,可以多个指针变量同时指向一个内存中的对象)。我们可以换手机(也就是让我们持有的指针指向另外一个手机),我们如果没有手机的时候,那么这个指针指向的就是一个NULL;我们想要拥有一个手机的前提是我们必须买一个(也就是为了持有该对象的前提是必须要手动创建)。同样,我们和这个对象的生命周期并没有绑定的关系,我们死了,手机还可以用,手机坏了,我们健康的活着。所以Delegation的关系就是一种生不带来死不带去的关系,不过和我们与东西之间的关系的区别在于,这个类里面有这个指针才有资格拥有这个东西的使用权,如果没有啊,那也是求不来的。这里面有个很大的好处就是,我们可以不断的换手机,而对于你的朋友来说他们知不知道你还没换没手机其实不重要,但对你来说,你可以换一台更好的手机,更漂亮的手机。搜易通过Delegation可以作为一个接口,背后通过他持有的指针进行实现所需要的功能,也就是相当于,朋友只要通过号码就能联系你,实现了联系的功能,但这个功能与你用哪部手机没多大关系,也就实现了不同功能的切换,这也就相当于(Handle/Body)的设计模式的一种现实生活的表现,非常方便的帮你完成了实现类的升级

  • 浅谈几种设计模式
    这些类与类之间的关系,构成了大量的设计模式,由于之后专门有课程讨论设计模式的问题,在这里简单使用图例来表示几种设计模式,来说明三种关系的使用场景。

    • Template Method(模版方法)

模版方法的UML

  • Observer模式

UML

  • Composite(组合模式)

Composite设计模式的UML
  • Prototype(原型模式)

Prototype的UML

这篇关于(Boolan) C++面向对象高级编程(三)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

MySQL高级查询之JOIN、子查询、窗口函数实际案例

《MySQL高级查询之JOIN、子查询、窗口函数实际案例》:本文主要介绍MySQL高级查询之JOIN、子查询、窗口函数实际案例的相关资料,JOIN用于多表关联查询,子查询用于数据筛选和过滤,窗口函... 目录前言1. JOIN(连接查询)1.1 内连接(INNER JOIN)1.2 左连接(LEFT JOI

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(