C++ 多重继承

Posted 我是小白呀

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++ 多重继承相关的知识,希望对你有一定的参考价值。

概述

多重继承 (multiple inheritance): 一个派生类有两个或多个基类, 派生类从两个或多个基类中继承所需的属性. C++ 为了适应这种情况, 允许一个派生类同时继承多个基类. 这种行为称为多重继承.
在这里插入图片描述

优缺点

优点

  • 自然地做到了对单继承的扩展
  • 可以继承多个类的功能

缺点

  • 结构复杂化
  • 优先顺序模糊
  • 功能冲突

声明多重继承的方法

格式

多重继承的格式:

派生类构造函数名(总形式参数表列): 
    基类1构造函数(实际参数表列),
    基类2构造函数(实际参数表列),
    基类3构造函数(实际参数表列)
{
    派生类中新增数成员据成员初始化语句
}

例子

Teacher 类:

#ifndef PROJECT5_TEACHER_H
#define PROJECT5_TEACHER_H

#include <string>
using namespace std;

class Teacher {
protected:
    string name;
    int age;
    string title;
public:
    Teacher(string n, int a, string t);
    void display_teacher();
};

#endif //PROJECT5_TEACHER_H

Teacher.cpp:

#include <iostream>
#include "Teacher.h"
using namespace std;

Teacher::Teacher(string n, int a, string t) : name(n), age(a), title(t) {}

void Teacher::display_teacher() {
    cout << "Teacher name: " << name << endl;
    cout << "age: " << age << endl;
    cout << "title: " << title << endl;
}

Student 类:

#ifndef PROJECT5_STUDENT_H
#define PROJECT5_STUDENT_H

#include <string>
using namespace std;

class Student {
protected:
    string name;
    char gender;
    double score;
public:
    Student(string n, char g, double s);
    void display_student();
};

#endif //PROJECT5_STUDENT_H

Student.cpp:

#include <iostream>
#include "Student.h"
using namespace std;

Student::Student(string n, char g, double s) : name(n), gender(g), score(s) {}

void Student::display_student() {
    cout << "Student name: " << name << endl;
    cout << "gender: " << gender << endl;
    cout << "score: " << score << endl;
}

Graduate 类:

#ifndef PROJECT5_GRADUATE_H
#define PROJECT5_GRADUATE_H

#include "Teacher.h"
#include "Student.h"
#include <string>
using namespace std;

class Graduate : public Teacher, public Student{
private:
    double wage;
public:
    Graduate(string t_n, int t_a, string t_t, string s_n, char s_g, double s_s);
    void display_graduate();
};

#endif //PROJECT5_GRADUATE_H

Graduate.cpp:

#include "Graduate.h"

Graduate::Graduate(string t_n, int t_a, string t_t, string s_n, char s_g, double s_s) :
    Teacher(t_n, t_a, t_t),
    Student(s_n, s_g, s_s) {}

void Graduate::display_graduate() {
    display_teacher();
    display_student();
}

main:

#include <iostream>
#include "Graduate.h"
using namespace std;

int main() {
    Graduate graduate1("王叔叔", 18, "隔壁老王", "我是小白呀", 'f', 99);
    graduate1.display_graduate();

    return 0;
}

输出结果:

Teacher name: 王叔叔
age: 18
title: 隔壁老王
Student name: 我是小白呀
gender: f
score: 99

二义性

二义性 (Ambiguity) 指在多重继承中, 两个基类中的数据成员名相同.
在这里插入图片描述
二义性在派生类中的解决方法:

  • 在标识符前用类名做前缀: Teacher::name 和 Student::name
  • 基类和派生类需要有一个完整的设计, 不能随意而为

两个基类有同名成员

在这里插入图片描述

A 类:

#ifndef PROJECT5_A_H
#define PROJECT5_A_H

#include <iostream>
using namespace std;

class A {
public:
    int num;
    void display() {cout << "A's num:" << num << endl;};
};

#endif //PROJECT5_A_H

B 类:

#ifndef PROJECT5_B_H
#define PROJECT5_B_H

#include <iostream>
using namespace std;

class B {
public:
    int num;
    void display() {cout << "B's num:" << num << endl;};
};

#endif //PROJECT5_B_H

C 类:

#ifndef PROJECT5_C_H
#define PROJECT5_C_H

#include <iostream>
#include "A.h"
#include "B.h"
using namespace std;

class C: public A, public B{
public:
    int c;
    void display() {cout << c << endl;};
};

#endif //PROJECT5_C_H

main:

#include <iostream>
#include "C.h"
using namespace std;

int main() {
    C c1;
    c1.A::num = 1;  // 用基类名限定
    c1.B::num = 2;  // 用基类名限定
    c1.A::display();
    c1.B::display();

    return 0;
}

输出结果:

A's num:1
B's num:2

错误的写法

#include <iostream>
#include "C.h"
using namespace std;

int main() {
    C c1;
    c1.num = 1;
    c1.display();
    
    return 0;
}

基类和派生类有同名成员

A 类:

class A {
public:
    int num;
    void display() {cout << "A's num:" << num << endl;};
};

B 类:

class B {
public:
    int num;
    void display() {cout << "B's num:" << num << endl;};
};

C 类:

class C: public A, public B{
public:
    int num;
    void display() {cout << "C's num:" << num << endl;};
};

main:

int main() {
    C c1;
    c1.num = 3;
    c1.A::num = 1;
    c1.B::num = 2;
    c1.display();
    c1.A::display();
    c1.B::display();

    return 0;
}

输出结果:

C's num:3
A's num:1
B's num:2

同名覆盖:

  • 基类的同名成员在派生类中被屏蔽, 成为 "不可见"的
  • 对成员函数, 限于函数名和参数个数相同, 类型相匹配. 若只有函数名相同而参数不同, 属于函数重载

两个基类从同一个基类派生

N 类:

class N {
public:
    int a;
    void display(){
        cout << "A::a=" << a <<endl;
    }
};

A 类:

class A : public N {
public:
    int a1;
};

B 类:

class B : public N {
public:
    int a2;
};

C 类:

class C: public A, public B{
public:
    int a3;
    void display() {cout << "a3=" << a3 << endl;};
};

main:

int main() {
    C c1;
    // 合法访问
    c1.A::a = 3;
    c1.A::display();

    return 0;
}

输出结果:

A::a=3

以上是关于C++ 多重继承的主要内容,如果未能解决你的问题,请参考以下文章

适配器模式(C++)多重继承和组合实现

第53课 被遗弃的多重继承

C++的多重继承

c++头脑风暴-多态虚继承多重继承内存布局

c++头脑风暴-多态虚继承多重继承内存布局

C++反汇编第四讲,认识多重继承,菱形继承的内存结构,以及反汇编中的表现形式.