(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++中std::placeholders的使用方法

《c++中std::placeholders的使用方法》std::placeholders是C++标准库中的一个工具,用于在函数对象绑定时创建占位符,本文就来详细的介绍一下,具有一定的参考价值,感兴... 目录1. 基本概念2. 使用场景3. 示例示例 1:部分参数绑定示例 2:参数重排序4. 注意事项5.

使用C++将处理后的信号保存为PNG和TIFF格式

《使用C++将处理后的信号保存为PNG和TIFF格式》在信号处理领域,我们常常需要将处理结果以图像的形式保存下来,方便后续分析和展示,C++提供了多种库来处理图像数据,本文将介绍如何使用stb_ima... 目录1. PNG格式保存使用stb_imagephp_write库1.1 安装和包含库1.2 代码解

C++实现封装的顺序表的操作与实践

《C++实现封装的顺序表的操作与实践》在程序设计中,顺序表是一种常见的线性数据结构,通常用于存储具有固定顺序的元素,与链表不同,顺序表中的元素是连续存储的,因此访问速度较快,但插入和删除操作的效率可能... 目录一、顺序表的基本概念二、顺序表类的设计1. 顺序表类的成员变量2. 构造函数和析构函数三、顺序表

使用C++实现单链表的操作与实践

《使用C++实现单链表的操作与实践》在程序设计中,链表是一种常见的数据结构,特别是在动态数据管理、频繁插入和删除元素的场景中,链表相比于数组,具有更高的灵活性和高效性,尤其是在需要频繁修改数据结构的应... 目录一、单链表的基本概念二、单链表类的设计1. 节点的定义2. 链表的类定义三、单链表的操作实现四、

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

深入解析Spring TransactionTemplate 高级用法(示例代码)

《深入解析SpringTransactionTemplate高级用法(示例代码)》TransactionTemplate是Spring框架中一个强大的工具,它允许开发者以编程方式控制事务,通过... 目录1. TransactionTemplate 的核心概念2. 核心接口和类3. TransactionT

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C

C++实现获取本机MAC地址与IP地址

《C++实现获取本机MAC地址与IP地址》这篇文章主要为大家详细介绍了C++实现获取本机MAC地址与IP地址的两种方式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实际工作中,项目上常常需要获取本机的IP地址和MAC地址,在此使用两种方案获取1.MFC中获取IP和MAC地址获取

C/C++通过IP获取局域网网卡MAC地址

《C/C++通过IP获取局域网网卡MAC地址》这篇文章主要为大家详细介绍了C++如何通过Win32API函数SendARP从IP地址获取局域网内网卡的MAC地址,感兴趣的小伙伴可以跟随小编一起学习一下... C/C++通过IP获取局域网网卡MAC地址通过win32 SendARP获取MAC地址代码#i

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(