码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • C++中菱形类关系再理解


    很多很多年前,还在北京读研时,我去 magnetar 寻求一个实习机会时,面我的北美小哥让我解释一下C++的菱形关系,好像当时没有答上来。

    今天在 OceanBase 代码里又看到了菱形关系。多重继承在我们的代码中是很少出现的,居然能再次遇到可真不容易。

    问题:下面的代码里,C 的构造函数为什么必须显式调用 Base(v)?

    class Base {  explicit A(int &) {} }
    class A : virtual public Base { }
    class B : virtual public Base { }
    class C :  public A, public B 
    {
       C(int v) : Base(v), A(), B() {}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    更进一步地,为什么下面比较简单的场景里 C 依然要显式调用 Base(v)?

    class Base {  explicit A(int &) {} }
    class A : virtual public Base { }
    class C :  public A
    {
       C(int v) : Base(v), A() {}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    Well,有了一些工作经验后会积累一些设计的思想,能够更好地站在设计者角度思考问题。回答上面两个问题就比较容易了:

    C++ 设计者引入了一条简单规则:A、B 这样使用了 virtual 继承的对象作为中间类时,没有资格去实例化基类。

    还是有点绕,更简单的思维:

    A、B 这样使用了 virtual 继承的对象作为中间类时,不会为A、B 分配基类对象的结构。

    只要不让他们去分配对象,那么自然就需要子类去分配对象,子类分配了对象,自然就需要主动调用 Base 的构造函数。

    C++设计者面对菱形问题时,解决思维如下:

    1. 要避免给基类(Base)分配多份内存
    2. 所以子类 A、B 都不应该去给基类(Base)分配内存
    3. 所以内存只能由 A、B 的子类 C 去分配内存
    4. 所以 C 需要调用构造函数。

    然后遇到一个特殊情况:

    1. 如果 Base 只有一个子类 A,A 只有一个子类 C,那么 A 能不能为 Base 分配内存呢?
    2. 这里如果考虑极致优化,此时可以让 A 给 Base 分配内存,C无需关注 Base。
    3. 但是这样做太特殊。所以设计者保持了一致的用户接口:Base 的内存必须由 C 分配。
  • 相关阅读:
    基于springboot在线考试报名系统毕业设计源码031706
    (18)线程的实例认识:线程的控制,暂停,继续,停止,线程相互控制,协作
    分布式数据库理论知识之CAP理论、ACID原则及分布式事务一致性算法
    重磅!首例植入配备 BCI 的神经刺激器的患者恢复手臂功能
    springboot毕设项目村庄管理系统9ype1(java+VUE+Mybatis+Maven+Mysql)
    OpenHD改造实现廉价高清数字图传(树莓派+PC)—(五)gstreamer视频采集、传输和显示
    LeetCode-882. Reachable Nodes In Subdivided Graph [C++][Java]
    Shell-基础(二):Shell变量、Shell运算符、Shell条件判断、Shell流程控制、函数
    深入理解Redis:工程师的使用指南
    Python识别图片的文字(Tesseract)和中文分词(jieba)
  • 原文地址:https://blog.csdn.net/maray/article/details/126646884
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号