C++ 多重继承


概述

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

C++ 多重继承_ios

优缺点

优点


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

缺点


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

声明多重继承的方法

格式

多重继承的格式:

派生类构造函数名(总形式参数表列): 
基类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) 指在多重继承中, 两个基类中的数据成员名相同.

C++ 多重继承_派生类_02

二义性在派生类中的解决方法:


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

两个基类有同名成员

C++ 多重继承_#include_03

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