C++ 基础

C++ 主页 C++ 概述 C++ 环境设置 C++ 基本语法 C++ 注释 C++ Hello World C++ 省略命名空间 C++ 标记 C++ 常量/字面量 C++ 关键字 C++ 标识符 C++ 数据类型 C++ 数字数据类型 C++ 字符数据类型 C++ 布尔数据类型 C++ 变量类型 C++ 变量作用域 C++ 多变量 C++ 基本输入/输出 C++ 修饰符类型 C++ 存储类 C++ 数字 C++ 枚举 C++ 枚举类 C++ 引用 C++ 日期和时间

C++ 运算符

C++ 运算符 C++ 算术运算符 C++ 关系运算符 C++ 逻辑运算符 C++ 位运算符 C++ 赋值运算符 C++ sizeof 运算符 C++ 条件运算符 C++ 逗号运算符 C++ 成员运算符 C++ 强制类型转换运算符 C++ 指针运算符 C++ 运算符优先级 C++ 一元运算符

C++ 控制语句

C++ 决策语句 C++ if 语句 C++ if else 语句 C++ 嵌套 if 语句 C++ switch 语句 C++ 嵌套 switch语句 C++ 循环类型 C++ while 循环 C++ for 循环 C++ do while 循环 C++ Foreach 循环 C++ 嵌套循环 C++ break 语句 C++ continue 语句 C++ goto 语句

C++ 字符串

C++ 字符串 C++ 循环遍历字符串 C++ 字符串长度 C++ 字符串连接 C++ 字符串比较

C++ 函数

C++ 函数 C++ 多函数参数 C++ 递归函数 C++ 返回值 C++ 函数重载 C++ 函数重写 C++ 默认参数

C++ 数组

C++ 数组 C++ 多维数组 C++ 指向数组的指针 C++ 将数组传递给函数 C++ 从函数返回数组

C++ 结构 &联合

C++ 结构 C++ 联合

C++ 指针

C++ 指针 C++ 解引用 C++ 修改指针

C++ 类和对象

C++ 面向对象 C++ 类 &对象 C++ 类成员函数 C++ 类访问修饰符 C++ 静态类成员 C++ 静态数据成员 C++ 静态成员函数 C++ 内联函数 C++ this 指针 C++ 友元函数 C++ 指向类的指针

C++ 构造函数

C++ 构造函数 &析构函数 C++ 默认构造函数 C++ 参数化构造函数 C++ 复制构造函数 C++ 构造函数重载 C++ 带默认参数的构造函数 C++ 委托构造函数 C++ 构造函数初始化列表 C++ 使用构造函数动态初始化

C++ 继承

C++ 继承 C++ 多重继承 C++ 多级继承

C++ 面向对象

C++ 重载 C++ 多态性 C++ 抽象 C++ 封装 C++ 接口 C++ 虚函数 C++ 纯虚函数与抽象类

C++ 文件处理

C++ 文件和流 C++ 文件读取

C++ 进阶

C++ 异常处理 C++ 动态内存 C++ 命名空间 C++ 模板 C++ 预处理器 C++ 信号处理 C++ 多线程 C++ Web 编程 C++ 套接字编程 C++ 并发 C++ 高级概念 C++ Lambda 表达式 C++ unordered_multiset

C++ 实用资源

C++ 问答 C++ 快速指南 C++ 速查表 C++ STL 教程 C++ 标准库 C++ 实用资源 C++ 讨论


C++ 中的多重继承

C++ 中的多重继承功能允许一个类从多个基类继承,这意味着派生类可以有多个父类,并从所有基类继承属性和行为。

实现多重继承

要实现多重继承,需要在派生类中指定多个基类,并使用逗号分隔的列表声明。

语法

C++ 中多重继承的语法为:-

class Base1 {
    // 基类 1 的成员
};

class Base2 {
    // 基类 2 的成员
};

class Derived : public Base1, public Base2 {
    // 派生类的成员
};

多重继承框图

请参阅下面的框图,演示多重继承 -

C++ 多重继承

根据上图,"Car"和"Boat"类是基类,它们派生自"DualModeVehicle"类,以实现多重继承。

多重继承示例

在下面的示例中,有两个基类:"Car"和"Boat",以及一个派生类"DualModeVehicle"。这两个基类都被派生类继承。

#include <iostream>
using namespace std;

// 基类 1
class Car {
   public:
      void drive() {
         cout << "Driving on land" << endl;
      }
};

// 基类 2
class Boat {
   public:
      void sail() {
         cout << "Sailing on water" << endl;
      }
};

// 派生类
class DualModeVehicle: public Car, public Boat {
   public:
      void use() {
        drive(); // 调用 Car 的 drive 函数
        sail(); // 调用 Boat 的 sail 函数
      }
};

int main() {
   DualModeVehicle myVehicle;
   myVehicle.use(); // 演示两种功能
   return 0;
}

输出

Driving on land
Sailing on water

解释

  • Car 类 是我们的第一个基类,它具有公共成员函数 drive(),而 boat 类 是第二个基类,它具有公共成员函数 sail()
  • 现在有一个名为 DualModeVehicle 的派生类,它继承自 CarBoat,并使用多重继承来组合两个基类的功能。
  • 它有一个公共成员函数 use(),它调用 Car 类中的 drive() 方法和 Boat 类中的 sail() 方法。

主函数

  • 现在这里,创建名为 myVehicleDualModeVehicle
  • 其中调用了 myVehicle 的 use() 方法,该方法又调用了 drive()sail()
  • 返回 0 表示执行成功。

多重继承中的挑战

C++ 中的多重继承允许一个类从多个基类继承,从而提供了灵活性和可重用性。然而,它也带来了一些挑战,如下所述 -

  • 歧义 - 当两个或多个基类具有相同名称的成员时,会导致歧义。
  • 钻石问题 - 当一个类继承自两个都继承自同一基类的类时,就会出现这种情况,由于基类的多个副本,会导致歧义和冲突,这最终被称为钻石问题。

多重继承中的歧义

如果两个或多个基类具有相同名称的成员(函数或变量),编译器将无法决定使用哪一个,最终导致歧义。

这可以通过作用域解析来解决。

语法

class Base1 {
public:
   void show() { cout << "Base1 show" << endl; }
};

class Base2 {
   public:
      void show() { cout << "Base2 show" << endl; }
};

class Derived : public Base1, public Base2 {
   public:
      void show() {
        Base1::show(); // 显式调用 Base1 的 show
        Base2::show(); // 显式调用 Base2 的 show
      }
};

处理多重继承中的歧义

这里我们将演示如何通过使用显式范围解析来指定应调用哪个基类的方法来处理多重继承中的歧义。

示例

让我们通过一个例子来思考

#include <iostream>
using namespace std;

class Base1 {
   public:
      void show() {
         cout << "Base1 show" << endl;
      }
};

class Base2 {
   public:
      void show() {
         cout << "Base2 show" << endl;
      }
};

class Derived : public Base1, public Base2 {
   public:
      void show() {
         // 这里出现歧义,因为 Base1 和 Base2 都有 show() 方法
         Base1::show(); // 明确调用Base1的节目
         Base2::show(); // 明确调用Base2的节目
      }
};

int main() {
   Derived obj;
   obj.show();  // 调用 Derived show() 方法,解决歧义
   return 0;
}

输出

Base1 show
Base2 show

在 int main() 函数体中,我们调用了 Deriveds 的 show() 方法,从而解决了歧义问题。

多重继承中的菱形问题

C++ 中的菱形问题是指一个类继承自两个类,而这两个类又都继承自同一个基类。由于派生类拥有该基类的两个副本,最终导致继承层次结构出现歧义,从而引发冲突。

示例

#include <iostream>
using namespace std;

class Base {
   public:
      void show() {
         cout << "Base show" << endl;
      }
};

class Derived1 : public Base {};
class Derived2 : public Base {};

class Final : public Derived1, public Derived2 {};

int main() {
    Final obj;
    // obj.show(); // 此行将导致歧义
    return 0;
}

多重继承中的钻石问题解决方案

C++ 中钻石问题的主要解决方案是使用虚拟继承

示例

#include <iostream>
using namespace std;

class Base {
   public:
      void show() {
         cout << "Base show" << endl;
      }
};

class Derived1 : virtual public Base {}; // 虚继承
class Derived2 : virtual public Base {}; // 虚继承

class Final : public Derived1, public Derived2 {};

int main() {
    Final obj;
    obj.show(); // 现在,这将毫无歧义地调用 Base 的 show() 方法
    return 0;
}

输出

Base show

通过使用虚拟继承,我们可以避免"钻石难题",从而确保派生类层次结构中只存在一个基类实例。

使用多重继承的好处

  • 代码可重用性,因为它允许开发人员使用现有类来创建具有组合功能的新类。
  • 它更准确地模拟现实世界中的实体,其中一个派生类可能具有多个基类的特征。
  • 它支持更灵活、更模块化的设计。