【C++】继承(万字详解) —— 切片 | 隐藏 | 子类的默认成员函数 | 菱形继承(下)

简介: 【C++】继承(万字详解) —— 切片 | 隐藏 | 子类的默认成员函数 | 菱形继承(下)

七. 菱形继承 & 菱形虚拟继承


🌈菱形继承


💠单继承:一个子类只有一个直接父亲

0a2653c851af460fa595bd959398a8f1.png2d65d23f6d4748949b924e4057485923.png

💠 多继承:一个子类有两个及两个以上的直接父亲


4cebaac233b3433da32a72337a77fc60.png


多继承看起来合理,其实就是坑,C++作为"第一个吃螃蟹的人"(Java后面的语言就避开了),带来了菱形继承,也就说助教对象中有两份Person,会有数据冗余和二义性的问题


二义性可以通过指定作用域勉强搞定


#include<string>
using namespace std;
class Person
{
public:
  string _name; //姓名
};
class Student : public Person
{
protected:
  int _stuid; //学号
};
class Teacher : public Person
{
protected:
  int _teacherid; //工号
};
class Assistant : public Student, public Teacher
{
protected:
  string _majorCourse; //主修课程
};
int main()
{
  //二义性:对_name的访问不明确
  Assistant a;
  //a._name = "peter"; //nope~ 错误示范,请勿模仿
  //需要显示指定访问哪个父类的成员
  a.Student::_name = "蛋哥";
  a.Teacher::_name = "杭哥"; 
  return 0;
}


那么数据冗余咋办呢?


🌈菱形虚拟继承


注意是在腰上添加virtual,不要乱用


class Person
{
public:
  string _name; //姓名
};
class Student : virtual public Person
{
protected:
  int _stuid; //学号
};
class Teacher :virtual public Person
{
protected:
  int _teacherid; //工号
};
class Assistant : public Student, public Teacher
{
protected:
  string _majorCourse; //主修课程
};
int main()
{
  Assistant a;
  //无需指定,访问的是同一个
  a._name = "蛋哥";
  a.Student::_name = "杭哥";
  a.Person::_name = "基哥";
  return 0;
}


🌈菱形虚拟继承的原理


为了研究原理:虚继承究竟是如何解决数据冗余的


class A
{
public:
  int _a;
};
class B : public A
//class B : virtual public A 
{
public:
  int _b;
};
class C : public A
//class C : virtual public A 
{
public:
  int _c;
};
class D : public B, public C
{
public:
  int _d;
};
int main()
{
  D d;
  d.B::_a = 1;
  d.C::_a = 2;
  d._b = 3;
  d._c = 4;
  d._d = 5;
  return 0;
}


先来实验不采用虚拟继承时,是有数据冗余 & 二义性问题的(且先继承的在前,后继承的在后) ——


0a2653c851af460fa595bd959398a8f1.png


💦 当观察虚拟继承时,可以观察到, A成员的确只存储了一份,在对象的最底下——

2d65d23f6d4748949b924e4057485923.png

此处我们发现:虚继承并没有节省空间,但是我们转换思路,a是一个int a[10000]数组,再那就节省了4万字节


但是B和C中是什么?推测是地址,众所周知,当前机器采取的是小端存储(低位存低地址,高位存高地址),我们再打开内存窗口来看这地址存的什么 ——


4cebaac233b3433da32a72337a77fc60.png


D对象中将A放到的了对象组成的最下面,这个A同时属于B和C,那么B和C如何去找到公共的A(虚基类)呢?就是通过了B和C的两个指针虚基表指针),指向的虚基表(找虚基类的表)。虚基表中存的偏移量,通过偏移量可以找到下面的A


Person关系菱形虚拟继承的原理 ——


6de278e6d6694ce5bb08e7e842b7e74b.png


A一般叫做虚基类,在D中,A放到一个公共位置,有时B需要找A、C需要找A,就要通过虚基表中的偏移量来计算。那为什么要找呢?考虑以下场景


//此时的B对象一部分是继承A的,一部分是自己的
    B d;
    B* Pb = &d; //切片,要找_a
    C c = d;
  C* pb = &d;


当我们求得bb的大小时候发现 ——


B bb;
  cout << sizeof(bb) << endl;
  bb._a = 1;
  bb._b = 2;

0a2653c851af460fa595bd959398a8f1.png

同时我们发现这种存储方式是十分巧妙的


void func(B* ptr)
{
  cout << ptr ->_a << endl;
}
func(&d);
func(&bb);


此处不知道ptr是指向父类还是子类的_a,但是父类和子类的存储结构都是保持一致的,_a都是放在最下面的位置。


注意:有公共祖先类就会构成菱形继承,那么virtual加在哪里呢?


0a2653c851af460fa595bd959398a8f1.png


注意虚拟继承是放在腰上,要解决的是A 的二义性和数据冗余问题


八. 总结


很多人说C++语法复杂,其实多继承就是一个体现。有了多继承,就存在菱形继承,有了菱形继承就有菱形虚拟继承,底层实现就很复杂。所以一般不建议设计出多继承,一定不要设计出菱形继承。否则在复杂度及性能上都有问题。多继承可以认为是C++的缺陷之一,很多后来的OO语言都没有多继承,如Java


💞继承和组合


继承和组合都是一种复用,只不过访问的方式有所不同,组合只能访问公有


// Car和BMW Car构成is-a的关系 
   //继承—— 白箱复用(white-box reuse)
   class Car{
   protected:
   string _colour = "白色"; // 颜色
   string _num = "粤A0DU95"; // 车牌号
   };
   class BMW : public Car{
   public:
   void Drive() {cout << "好开-操控" << endl;}
   };


// Tire和Car构成has-a的关系(轮胎和车)
   //组合 - 黑箱复用(black-box reuse)
   class Tire{
   protected:
       string _brand = "Michelin";  // 品牌
       size_t _size = 17;         // 尺寸
   };
   class Car{
   protected:
   string _colour = "白色"; // 颜色
   string _num = "粤A0DU95"; // 车牌号
    Tire _t; // 轮胎
   };


public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象

组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象

如果既符合继承和组合,优先使用对象组合,而不是类继承

继承允许你根据基类的实现来定义派生类的实现。这种通过生成派生类的复用通常被称为白箱复用(white-box reuse)。术语“白箱”是相对可视性而言:在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高

对象组合是类继承之外的另一种复用选择。新的更复杂的功能可以通过组装或组合对象来获得。对象组合要求被组合的对象具有良好定义的接口。这种复用风格被称为黑箱复用(black-box reuse),因为对象的内部细节是不可见的。对象只以“黑箱”的形式出现。组合类之间没有很强的依赖关系,耦合度低。优先使用对象组合有助于你保持每个类被封装

实际尽量多去用组合。组合的耦合度低,代码维护性好。不过继承也有用武之地的,有些关系就适合继承那就用继承,另外要实现多态,也必须要继承。类之间的关系可以用继承,可以用组合,就用组合

另外我们希望,模块之间的关系最好是 低耦合高内聚,方便维护。继承(A和B),耦合度高,依赖性强,任意一个成员的改变都会对我有影响;组合(C和D),耦合度低,依赖关系较弱 (继承就是跟团游 ,组合就是自由行)


结论:完全符合is-a,就用继承;完全符合has-a,就用组合;既是is-a,又是has-a,优先用组合而不是继承


九. 常见面试题


1️⃣如何定义一个不能被继承的类?

父类构造函数私有 —— 子类是不可见;子类对象实例化,无法调用构造函数

C++11 final关键字

//方法一:父类构造函数私有
class A 
{
private:
  A()
  {}
protected:
  int _a;
};
class B : public A
{
};
int main()
{
  B bb;
  return 0;
}


方法二:final关键字


0a2653c851af460fa595bd959398a8f1.png


2️⃣ 多继承中指针偏移问题

下面说法正确的是( )


class Base1 {  public:  int _b1; };
class Base2 {  public:  int _b2; };
class Derive : public Base1, public Base2 { public: int _d; };
int main(){
  Derive d;
  Base1* p1 = &d;
   Base2* p2 = &d;
   Derive* p3 = &d;
   return 0;
}


A:p1 == p2 == p3

B:p1 < p2 < p3

C:p1 == p3 != p2

D:p1 != p2 != p3

E:编译报错

F:运行报错


0a2653c851af460fa595bd959398a8f1.png


答案选C


相关文章
|
3月前
|
存储 安全 Java
c++--继承
c++作为面向对象的语言三大特点其中之一就是继承,那么继承到底有何奥妙呢?继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用,继承就是类方法的复用。
79 0
|
3月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
72 0
|
6月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
307 6
|
8月前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
171 5
|
7月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
3月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
154 0
|
5月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
152 12
|
6月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
120 16
|
7月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
6月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。