【C++】继承(下)

简介: 【C++】继承(下)

5. 继承与友元


对于友元关系,类继承的时候是不能被继承的,也就是说,基类友元不能访问派生类的私有和保护成员,如果需要访问的话,就需要在派生类中也规定友元。

class Student;
class Person
{
public:
  friend void Display(const Person& p, const Student& s);
protected:
  string _name; // 姓名
};
class Student : public Person
{
protected:
  int _stuNum; // 学号
};
void Display(const Person& p, const Student& s)
{
  cout << p._name << endl;
  cout << s._stuNum << endl;
}
void Test6()
{
  Person p;
  Student s;
  Display(p, s);
}

82af5f23f24c007e24614ca15e9017f1.png

解决方案:

0fcb2c800e05f93b9a4225faa4a98fa9.png


6. 继承与静态成员


在基类中定义了一个静态成员,则此成员在继承体系中是共享的,只有一个这样的成员

class Person
{
public:
  Person() { ++_count; }
protected:
  string _name; // 姓名
public:
  static int _count; // 统计人的个数。
};
int Person::_count = 0;
class Student : public Person
{
protected:
  int _stuNum; // 学号
};
class Graduate : public Student
{
protected:
string _seminarCourse; // 研究科目
};
void Test7()
{
  Student s1;
  Student s2;
  Student s3;
  Graduate s4;
  cout << "人数:" << Person::_count << endl;
  Student::_count = 0;
  cout << "人数:" << Person::_count << endl;
}

8335e9d1079528c6d764e00392ce5ead.png


7. 菱形继承与菱形虚拟继承


1. 继承的种类

在C++继承的设计中,为了满足各种实际情况,设计了单继承多继承

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

多继承:一个子类有超过一个父类

3dcc643b78fde3db377c819fb4aa4cc2.png

在上述情况的前提下,会出现一种可能的继承结构:菱形继承

18c9062ddf80002ec3aa7b0af46f2a5c.png

在菱形继承中就会出现数据冗余和二义性的问题。对于上述的例子,在Assistant对象中Person的成员会有两份,造成了数据冗余,如果在Assistant对象中访问Person中的成员就会出现二义性,不知道访问Student基类还是Teacher基类中的Person成员。虽然我们可以使用显示指定访问类的方法解决二义性的问题,但是对于数据冗余还是没有办法解决,所以这里引入了虚拟继承的方式


2. 虚拟继承

对于上面的继承关系,在Student和Teacher的继承中使用Person虚拟继承,就可以解决问题,但是这里注意虚拟继承不要在其他地方使用

class Person
{
public:
  string _name; // 姓名
};
//class Student : public Person
class Student : virtual public Person//虚拟继承
{
protected:
  int _num; //学号
};
//class Teacher : public Person
class Teacher : virtual public Person//虚拟继承
{
protected:
  int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:
  string _majorCourse; // 主修课程
};


虚拟继承解决数据冗余和二义性的原理

首先,这里我们给出一个简化的菱形继承的示例

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;
};
void Test8()
{
  D d;
  d.B::_a = 1;
  d.C::_a = 2;
  d._b = 3;
  d._c = 4;
  d._d = 5;
}

如果不适用虚拟继承的方式,那么对象d的内容是这样的

b7e2f1b66ce8be9001ae95febc40ebb0.png

当使用虚拟继承之后,d对象的内容是这样的:

9614fc4dafe148823063554aa9a856a7.png

可以看到,这里是把公共的_a放到了最下面,这个_a同时属于B和C,那么他们怎么找到_a呢?这里是通过B和C的两个指针,指向的一张虚基表,着两个指针叫做虚基表指针,虚基表指针中存放的是偏移量,通过偏移量就可以找到下面的_a。

对上述的结构,有下面一张图来辅助理解:

57fded429ba69bec4f034d11500b1227.png


8. 继承与组合


继承的本质就是类的复用,类的复用除了继承之外,还有一种复用方式叫组合,二者最大的区别就是访问方式不同

  • public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。
  • 组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象
class M
{
};
class N : public M//public继承
{
};
class X//组合
{
public:
  M mm;
};


继承实际上是一种白箱复用,组合是一种黑箱复用,继承中派生类对于基类的内部细节大多可见,一定程度上破坏了封装,而组合就不会关心内部实现,只是通过提供的接口来使用,高内聚,低耦合,有很好的封装性。当然,由于组合使用的特性,对被组合类的接口要求就相对高一点。


综上,这里有一个原则:优先使用对象组合,而不是继承。


9.总结


  1. 很多人说C++语法复杂,其实多继承就是一个体现。有了多继承,就存在菱形继承,有了菱形继承就有菱形虚拟继承,底层实现就很复杂。所以一般不建议设计出多继承,一定不要设计出菱形继承。否则在复杂度及性能上都有问题。


  1. 多继承可以认为是C++的缺陷之一,很多后来的OO语言都没有多继承,如Java。


  1. 在继承和组合中,优先使用组合优先使用对象组合,而不是类继承。
相关文章
|
2月前
|
安全 Java 编译器
C++进阶(1)——继承
本文系统讲解C++继承机制,涵盖继承定义、访问限定符、派生类默认成员函数、菱形虚拟继承原理及组合与继承对比,深入剖析其在代码复用与面向对象设计中的应用。
|
6月前
|
存储 安全 Java
c++--继承
c++作为面向对象的语言三大特点其中之一就是继承,那么继承到底有何奥妙呢?继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用,继承就是类方法的复用。
148 0
|
9月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
477 6
|
11月前
|
C++ 开发者
C++学习之继承
通过继承,C++可以实现代码重用、扩展类的功能并支持多态性。理解继承的类型、重写与重载、多重继承及其相关问题,对于掌握C++面向对象编程至关重要。希望本文能为您的C++学习和开发提供实用的指导。
166 16
|
11月前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
326 5
|
编译器 C++ 开发者
【C++】继承
C++中的继承是面向对象编程的核心特性之一,允许派生类继承基类的属性和方法,实现代码复用和类的层次结构。继承有三种类型:公有、私有和受保护继承,每种类型决定了派生类如何访问基类成员。此外,继承还涉及构造函数、析构函数、拷贝构造函数和赋值运算符的调用规则,以及解决多继承带来的二义性和数据冗余问题的虚拟继承。在设计类时,应谨慎选择继承和组合,以降低耦合度并提高代码的可维护性。
238 1
【C++】继承
|
安全 程序员 编译器
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
202 11
|
C++
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
161 1
|
C++
C++番外篇——虚拟继承解决数据冗余和二义性的原理
C++番外篇——虚拟继承解决数据冗余和二义性的原理
153 1
C++(二十)继承
本文介绍了C++中的继承特性,包括公有、保护和私有继承,并解释了虚继承的作用。通过示例展示了派生类如何从基类继承属性和方法,并保持自身的独特性。此外,还详细说明了派生类构造函数的语法格式及构造顺序,提供了具体的代码示例帮助理解。