C++多继承,虚继承部分总结与示例
tags: C++ OOP
写在前面
写一下多继承, 虚继承的一些部分, 包括一些例子.
多继承
简介
多继承是指从多个直接基类中产生派生类的能力. 多继承的派生类继承了所有父类的属性, 所以会带来一些复杂的问题.
示例1: 多继承用法与调用顺序
#include <string>
#include <iostream>
using namespace std;class ZooAnimal {
public:ZooAnimal() { cout << "call ZooAnimal::ZooAnimal()\n"; }~ZooAnimal() { cout << "call ZooAnimal::~ZooAnimal()\n"; }
};class Endangered {
public:Endangered() { cout << "call Endangered::Endangered()\n"; }Endangered(int a) : m_a(a) { cout << "call Endangered::Endangered(int)\n"; }~Endangered() { cout << "call Endangered::~Endangered()\n"; }static int critical;private:int m_a;
};
int Endangered::critical = 10;class Bear : public ZooAnimal {
public:Bear() { cout << "call Bear::Bear()\n"; }Bear(string, bool, string);~Bear() { cout << "call Bear::~Bear()\n"; }
};
Bear::Bear(string name, bool onExhibit, string detail) {cout << "call Bear::Bear(string, bool, string)\n";
}// multi inherit
class Panda : public Bear, public Endangered {
public:Panda();Panda(string, bool);~Panda() { cout << "call Panda::~Panda()\n"; }
};Panda::Panda(string name, bool onExhibit): Bear(name, onExhibit, "Panda"), Endangered(Endangered::critical) {cout << "call Panda::Panda(string, bool)\n";
}
Panda::Panda() : Endangered(Endangered::critical) {cout << "call Panda::Panda()\n";
}void t1() {//Panda p1;/* call ZooAnimal::ZooAnimal() *//* call Bear::Bear() *//* call Endangered::Endangered(int) *//* call Panda::Panda() *//* call Panda::~Panda() *//* call Endangered::~Endangered() *//* call Bear::~Bear() *//* call ZooAnimal::~ZooAnimal() */
}int main(int argc, char *argv[]) {t1();return 0;
}
通过输出可以看出, 首先调用最终基类, 然后调用直接基类, 然后是第二基类, 最后是子类.
析构顺序正好相反(由子类至基类).
示例2: 多继承构造函数可能出现的问题
#include <iostream>
#include <memory>
#include <string>
using namespace std;struct Base1 {Base1() { cout << "Base1()\n"; }Base1(const string &) { cout << "Base1(const string&) \n"; }Base1(std::shared_ptr<int>);
};struct Base2 {Base2() { cout << "Base2()\n"; }Base2(const string &) { cout << "Base2(const string&) \n"; }Base2(int a) { cout << "Base2(int)\n"; }
};struct D1 : public Base1, public Base2 {using Base1::Base1;using Base2::Base2;// 定义自己版本的构造函数D1(const string &) { cout << "D1(const string &)\n"; }
};void t1() {D1 d1(1);/* Base1() *//* Base2(int) */
}
void t2() {D1 d2("abc");/* error: call of overloaded 'D1(const char [4])' is ambiguous *//* Base1() *//* Base2() *//* D1(const string &) */
}int main(int argc, char *argv[]) {// test/* t1(); */t2();return 0;
}
多继承引发的类型转换问题
在只有一个基类的情况下, 派生类的指针或引用能自动转换成一个可访问基类的指针或引用, 多基类情况类似, 可以令某个可访问基类的指针或引用直接指向一个派生类对象. 如下所示:(仍然采用上面的Panda例子)
void print(const Bear &) { cout << "call print(const Bear&)\n"; }
void highlight(const Endangered &) {cout << "call highlight(const Endangered&)\n";
}
ostream &operator<<(ostream &os, const ZooAnimal &) {os << "call operator<< (ZooAnimal)\n";return os;
}
// 如果解注释, 会导致错误
/* error: call of overloaded 'print(Panda&)' is ambiguous */
/* void print(const Endangered &) { cout << "call print(const Endangered&)\n"; } */
void t2() {Panda aa("aa", true);print(aa);highlight(aa);cout << aa << endl;/* call print(const Bear&) *//* call highlight(const Endangered&) *//* call operator<< (ZooAnimal) */
}
上面如果为同一函数的参数作两个基类的重载, 那么就会导致二义性错误.
基于指针类型或引用类型的查找
与只有一个基类的继承一样, 对象/指针/引用的静态类型决定了能使用的成员.
例如: 如果使用一个ZooAnimal指针, 则只有定义在ZooAnimal中的操作可以调用, 而Pandas中的其他特有部分(其他基类, Bear,Panda,Endangered)都不可见.
class ZooAnimal {
public:ZooAnimal() { cout << "call ZooAnimal::ZooAnimal()\n"; }void print() { cout << "call ZooAnimal::print()\n"; }~ZooAnimal() { cout << "call ZooAnimal::~ZooAnimal()\n"; }
};void t3() {Bear *pb = new Panda("aa", true);pb->print();delete pb;/* call ZooAnimal::ZooAnimal() *//* call Bear::Bear(string, bool, string) *//* call Endangered::Endangered(int) *//* call Panda::Panda(string, bool) *//* call ZooAnimal::print() *//* call Bear::~Bear() *//* call ZooAnimal::~ZooAnimal() */
}
多继承下的类作用域
在只有一个基类的情况下, 派生类的作用域嵌套在直接基类和间接基类的作用域中. 查找过程沿着继承体系自底向上进行, 直到找到所需的名字. 派生类的名字将隐藏基类的同名成员.
多继承中, 相同的查找过程在所有直接基类中同时进行, 若同一名字在多个基类中找到, 则二义性错误.
所以, 需要显式指明作用域, 或者定义新版本的不同名内容.
虚继承
派生列表中同一基类只能出现一次, 但是实际上派生类可以多次继承同一个类, 派生类可以通过它的两个直接基类分别继承同一个间接基类, 也可以直接继承某个基类, 然后通过另一个基类再一次间接继承该类.
默认情况下, 派生类中含有继承链上的每一个类对应的子部分, 如果某个类在派生过程中出现多次, 则派生类中将包含该类的多个子对象.
虚继承的出现就是用于解决多继承中存在的基类多次使用问题的.
其目的是: 令某一个类作出声明, 承诺愿意共享基类, 其中, 共享的基类子对象称为虚基类.
在这种机制下, 不论虚基类在继承体系中出现了多少次, 在派生类中都只包含唯一一个共享的虚基类子对象.
- 必须在虚派生的真实需求出现之前完成派生操作.
- 虚派生只会影响从指定了虚基类的派生类中进一步派生出的类, 不会影响派生类本身.
- 使用
virtual说明符(在public之前或者之后都可以)表明了: 在后续的派生类中共享虚基类的同一份实例, 但是并没有规定什么样的类能够作为虚基类.
示例: Panda
#include <string>
#include <iostream>
using namespace std;class ZooAnimal {
public:ZooAnimal() {}void print() { cout << "call ZooAnimal::print()\n"; }~ZooAnimal() {}
};class Endangered {
public:Endangered() {}Endangered(int a) : m_a(a) {}~Endangered() {}static int critical;private:int m_a;
};
int Endangered::critical = 10;class Raccoon : virtual public ZooAnimal {
public:Raccoon() {}Raccoon(string, bool, string);~Raccoon() {}
};
Raccoon::Raccoon(string name, bool onExhibit, string detail) {}class Bear : virtual public ZooAnimal {
public:Bear() {}Bear(string, bool, string);~Bear() {}
};
Bear::Bear(string name, bool onExhibit, string detail) {}// multi inherit
class Panda : public Bear, public Raccoon, public Endangered {
public:Panda();Panda(string, bool);~Panda() {}
};Panda::Panda(string name, bool onExhibit): Bear(name, onExhibit, "Panda"), Endangered(Endangered::critical) {}
Panda::Panda() : Endangered(Endangered::critical) {}void t1() {Panda a;/* call ZooAnimal::ZooAnimal() *//* call Bear::Bear() *//* call Raccoon::Raccoon() *//* call Endangered::Endangered(int) *//* call Panda::Panda() *//* call Panda::~Panda() *//* call Endangered::~Endangered() *//* call Raccoon::~Raccoon() *//* call Bear::~Bear() *//* call ZooAnimal::~ZooAnimal() */
}
虚派生中, 虚基类是由最底层的派生类初始化的, 以上面的程序为例, 当创建Panda对象时, 由Panda构造函数独自控制ZooAnimal的初始化过程.
支持向基类的常规类型转换
与非虚基类一样, 派生类对象也可以被可访问基类的指针或引用操作.
void dance(const Bear &) { cout << "call dance(const Bear&)\n"; }
void rummage(const Raccoon &) { cout << "call rummage(const Raccoon&)\n"; }
ostream &operator<<(ostream &os, const ZooAnimal &) {os << "call operator<< (ZooAnimal)\n";return os;
}void t2() {Panda a;dance(a);rummage(a);cout << a;/* call dance(const Bear&) *//* call rummage(const Raccoon&) *//* call operator<< (ZooAnimal) */
}
虚基类成员的可见性
因为在每一个共享的虚基类中只有唯一一个共享的子对象, 所以该基类的成员可以被直接访问, 并且不会产生二义性.
如果虚基类的成员只被一条派生路径覆盖, 则我们仍可以直接访问这个被覆盖的成员, 但是如果成员被多于一个的基类覆盖(例子: 菱形继承), 派生类就必须为该成员自定义一个新版本.
虚继承对象的构造方式
- 首先使用提供给最底层派生类构造函数的初始值初始化该对象的虚基类子部分;
- 接下来按照直接基类在派生列表中出现的次序依次对其初始化.
例子中的顺序:
- 使用Panda的构造函数初始值列表中提供的初始值构造虚基类ZooAnimal部分
- 构造Bear
- 构造Raccoon
- 构造Endangered
- 构造Panda
如果Panda没有显式初始化ZooAnimal基类, 则调用ZooAnimal的默认构造函数, 所以此时一定要有默认构造函数.
相关文章:
C++多继承,虚继承部分总结与示例
tags: C OOP 写在前面 写一下多继承, 虚继承的一些部分, 包括一些例子. 多继承 简介 多继承是指从多个直接基类中产生派生类的能力. 多继承的派生类继承了所有父类的属性, 所以会带来一些复杂的问题. 示例1: 多继承用法与调用顺序 #include <string> #include <…...
程序员35岁以后就没有出路了吗?听听京东10年测开的分析
国内的互联网行业发展较快,所以造成了技术研发类员工工作强度比较大,同时技术的快速更新又需要员工不断的学习新的技术。因此淘汰率也比较高,超过35岁的基层研发类员工,往往因为家庭原因、身体原因,比较难以跟得上工作…...
数据结构(六):冒泡排序、选择排序、插入排序、希尔排序、快速排序
数据结构(六)一、大O表示法二、冒泡排序三、选择排序一、大O表示法 在计算机中采用粗略的度量来描述计算机算法的效率,这种方法被称为“大O”表示法。 我们判断一个算法的效率,不能只凭着算法运行的速度,因为随着数据…...
C++之类与对象(上)
目录 一、类的定义 二.类的访问限定及封装 1.访问限定 2.封装 三.类的作用域和实例化 2.类的实例化 四.类的对象大小的计算 1.类成员存储方式 2.结构体内存对齐规则 五.类成员函数的this指针 1.this指针的引出 2.this指针的特性 3.C语言和C实现Stack的对比 一、类的定义 class …...
Java岗面试题--Java并发 计算机网络(日积月累,每日三题)
目录1. 面试题一:在 Java 程序中怎么保证多线程的运行安全?1.1 追问一:Java 线程同步的几种方法?2. 面试题二:JMM3. 面试题三:计算机网络的各层协议及作用?1. 面试题一:在 Java 程序…...
三菱FX3U与威纶MT8071IP走RS422通讯
一、准备工作 1.需要工具: 电脑一台、PLC:三菱FX3U一个、触摸屏:威纶MT8071一个、 (三菱圆形编程口转USB)一根、触摸屏与电脑通讯线一根(T型口数据线)、PLC与触摸屏通讯线:电烙…...
给想考CISP的一点建议
如果你正在考虑参加CISP认证考试,以下是我对你的几点建议: 了解CISP考试: 在报名参加考试之前,要充分了解CISP认证考试的考试内容、考试形式、考试难度等相关信息,这有助于你制定更有效的备考计划。制定备考计划&…...
ACM 记忆化搜索
一.记忆化搜索概述 1.概念 搜索是一种简单有效但是效率又很低下的算法结构,其低效的原因主要在于存在很多重叠子问题。而记忆化搜索则是在搜索的基础上,利用数组来记录已经计算出来的重叠子问题状态,进行合理化的剪枝,从而降低时…...
spring框架常用注解简单说明
1、Configuration:标注在类上,相当于把当前类作为spring的xml配置文件中的; 2、Bean:标注在方法上,相当于spring配置文件中的; 3、Service:标注在类上,表明当前类是一个服务层的Be…...
2023-02-24 mysql/innodb-聚合-临时表避免OOM-使用磁盘文件-分析
摘要: mysql/innodb在执行聚合时, 当聚合的数据量太大时, 也就是临时表的大小超过tmp_table_size 限制时, 将进行写磁盘操作, 以避免OOM。 本文记录聚合数据写磁盘的操作。 参考: https://dev.mysql.com/doc/refman/5.7/en/server-system-variables.html#sysvar_tmp_table_…...
cracklib与libpwquality 评估密码的安全性
一、cracklib 检测密码强弱linux中采用pam pam_cracklib module来实现对密码强度的检测,可以通过配置让linux系统自动检测用户的密码是否为弱密码。yuminstall cracklib # centos apt-get install libcrack2 # ubuntu # 如果需要依赖此库做开发的话需要安装这个 y…...
【Java】保证并发安全的三大特性
一、并发编程三大特性的定义和由来 并发编程这三大特性就是为了在多个线程交替执行任务的过程中保证线程安全性。 二、为什么会出现线程不安全的现象呢? 接下来我们从这三个特性切入来介绍线程不安全的原因。 1.原子性: 一组操作要么全部执行&#…...
如何优雅的用golang封装配置项(Functional Options)
导读 最近要封装一个公共服务,涉及到配置项的地方总是找不到合理的方案,后来看了一下grpc在配置方面的封装,了解到原来是golang特有的Functional Options编程模式,今天分享给大家,希望你能用到,咱们直接来看…...
Springboot 使用thymeleaf 服务器无法加载resources中的静态资源异常处理
目录一、异常错误二、原因三、解决方法方法1. 将无法编译的静态资源放入可编译目录下方法2. 重新编译项目加载资源方法3. 修改pom.xml资源配置文件方法4. 不连接远程数据库启动,使用本地数据库一、异常错误 Springboot使用thymeleaf,并连接远程数据库启…...
服务端IOS订阅类型支付接入详细说明与注意事项
一、说明 由于本人在开发ios订阅类型支付接入的时候,遇到了很多坑,也查了不少资料,逐步完善了整个ios订阅支付服务端接入的功能,在这里写下总结和一些注意事项的记录,方便未来需要重新接入或者避免一些不必要的坑,这里…...
【剑指Offer】重建二叉树(递归+迭代)
重建二叉树一、递归法二、迭代法题目链接 题目描述: 输入某二叉树的前序遍历和中序遍历的结果,请构建该二叉树并返回其根节点。 假设输入的前序遍历和中序遍历的结果中都不含重复的数字。 示例 1: Input: preorder [3,9,20,15,7], inorder [9,3,15,…...
注解@Transactional 原理和常见的坑
这篇文章,会先讲述 Transactional 的 4 种不生效的 Case,然后再通过源码解读,分析 Transactional 的执行原理,以及部分 Case 不生效的真正原因1 项目准备下面是 DB 数据和 DB 操作接口:uidunameusex1张三女2陈恒男3楼仔…...
2023年全国最新交安安全员精选真题及答案4
百分百题库提供交安安全员考试试题、交安安全员考试预测题、交安安全员考试真题、交安安全员证考试题库等,提供在线做题刷题,在线模拟考试,助你考试轻松过关。 31.特种劳动防护用品必须具有“三证”,下列不属于“三证”的是&#…...
扬帆优配|半天翻倍,“蹭热点”翻车,前期“牛股”已近腰斩
周五上午,A股商场整体走低,多数职业板块和个股跌落,军工和核算机等板块逆势上涨,北向资金半天净卖出额约38亿元。 个股方面,昨夜公告被证监会立案查询的奥联电子股价再度大跌,盘中最贱价较近期高位已腰斩。…...
6 种易于上手的编程副业,每月赚取 1,000 多美元——没有废话
没有自由职业者或博客,也不需要前期费用。你们中的大多数人阅读这样的故事是希望其中的一些故事能帮助您赚更多的钱。好吧,几年前我还是同一个人。我希望尝试一些新的副业并赚点钱。其中一个视频建议我在网上写作,此后我写了很多技术文章。在…...
QMC5883L的驱动
简介 本篇文章的代码已经上传到了github上面,开源代码 作为一个电子罗盘模块,我们可以通过I2C从中获取偏航角yaw,相对于六轴陀螺仪的yaw,qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
Java入门学习详细版(一)
大家好,Java 学习是一个系统学习的过程,核心原则就是“理论 实践 坚持”,并且需循序渐进,不可过于着急,本篇文章推出的这份详细入门学习资料将带大家从零基础开始,逐步掌握 Java 的核心概念和编程技能。 …...
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南 在数字化营销时代,邮件列表效度、用户参与度和网站性能等指标往往决定着创业公司的增长成败。今天,我们将深入解析邮件打开率、网站可用性、页面参与时…...
推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材)
推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材) 这个项目能干嘛? 使用 gemini 2.0 的 api 和 google 其他的 api 来做衍生处理 简化和优化了文生图和图生图的行为(我的最主要) 并且有一些目标检测和切割(我用不到) 视频和 imagefx 因为没 a…...
QT3D学习笔记——圆台、圆锥
类名作用Qt3DWindow3D渲染窗口容器QEntity场景中的实体(对象或容器)QCamera控制观察视角QPointLight点光源QConeMesh圆锥几何网格QTransform控制实体的位置/旋转/缩放QPhongMaterialPhong光照材质(定义颜色、反光等)QFirstPersonC…...
C++.OpenGL (20/64)混合(Blending)
混合(Blending) 透明效果核心原理 #mermaid-svg-SWG0UzVfJms7Sm3e {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-icon{fill:#552222;}#mermaid-svg-SWG0UzVfJms7Sm3e .error-text{fill…...
【C++特殊工具与技术】优化内存分配(一):C++中的内存分配
目录 一、C 内存的基本概念 1.1 内存的物理与逻辑结构 1.2 C 程序的内存区域划分 二、栈内存分配 2.1 栈内存的特点 2.2 栈内存分配示例 三、堆内存分配 3.1 new和delete操作符 4.2 内存泄漏与悬空指针问题 4.3 new和delete的重载 四、智能指针…...
