深入探索C++继承机制:从概念到实践的全面指南

目录

继承的概念及定义

  继承的概念

  继承的定义

        定义格式

        继承方式和访问限定符

        继承基类成员访问方式的变化

        默认继承方式

基类和派生类对象赋值转换

继承中的作用域

派生类的默认成员函数

继承与友元

继承与静态成员

继承的方式

菱形虚拟继承

菱形虚拟继承原理

继承的总结和反思

相关笔试面试题


继承的概念及定义

  继承的概念

        C++中的继承是面向对象编程的一个核心特性,它允许创建一个新类(派生类或子类)基于已存在的类(基类或父类)的结构和行为。

        继承的主要目的是实现代码的复用和促进软件的模块化设计。

代码举例:

        以下代码中Student类和Teacher类继承了Person类。 

//父类
class Person
{
public:
	void Print()
	{
		cout << "name:" << _name << endl;
		cout << "age:" << _age << endl;
	}
protected:
	string _name = "张三"; //姓名
	int _age = 18;     //年龄
};

//子类
class Student : public Person
{
protected:
	int _stuid;   //学号
};

//子类
class Teacher : public Person
{
protected:
	int _jobid;   //工号
};

        继承后,父类Person的成员,包括成员函数和成员变量,都会变成子类的一部分,也就是说,子类Student和Teacher复用了父类Person的成员。 

  继承的定义

        定义格式

class 派生类名 : 访问修饰符 基类名 
{
    // 派生类的成员定义
};

其中:

  • 派生类名是要定义的新类的名称。
  • 访问修饰符publicprotectedprivate,用于指定基类成员在派生类中的访问权限。
  • 基类名是已经被定义的、要从中继承的类的名称。

        继承方式和访问限定符

        我们知道,访问限定符有以下三种:

  1. public访问
  2. protected访问
  3. private访问

        而继承的方式也有类似的三种:

  1. public继承
  2. protected继承
  3. private继承

        继承基类成员访问方式的变化

        基类当中被不同访问限定符修饰的成员,以不同的继承方式继承到派生类当中后,该成员最终在派生类当中的访问方式将会发生变化。

类成员/继承方式public继承protected继承private继承
基类的public成员派生类的public成员派生类的protected成员派生类的private成员
基类的protected成员派生类的protected成员派生类的protected成员派生类的private成员
基类的private成员在派生类中不可见在派生类中不可见在派生类中不可见

        实际上基类成员访问方式的变化规则也不是无迹可寻的,我们可以认为三种访问限定符的权限大小为:public > protected > private,基类成员访问方式的变化规则如下:

  1. 公有继承(public inheritance):

    • 基类的公有成员在派生类中保持公有。
    • 基类的保护成员在派生类中变为保护成员。
    • 基类的私有成员在派生类中不可直接访问,但会影响派生类的接口(比如通过基类的公有或保护成员函数间接访问)。
  2. 保护继承(protected inheritance):

    • 基类的公有和保护成员在派生类中都变为保护成员。
    • 基类的私有成员在派生类中同样不可直接访问,但可能影响派生类的内部实现。
  3. 私有继承(private inheritance):

    • 不论基类成员的原始访问权限如何,它们在派生类中都变为私有成员。
    • 这意味着基类的公有和保护成员在派生类的外部都是不可见的,且派生类的子类也不能访问这些成员。

        虽然Student类继承了Person类,但是我们无法在Student类当中访问Person类当中的private成员_name。 

代码举例:

#include <string>
#include <iostream>

using namespace std;

//基类
class Person
{
private:
	string _name = "张三"; //姓名
};

//派生类
class Student : public Person
{
public:
	void Print()
	{
		//在派生类当中访问基类的private成员,error!
		cout << _name << endl;
	}
protected:
	int _stuid;   //学号
};

代码结果:

        基类的private成员在派生类中是不可直接访问的,无论是从派生类的内部还是外部。这些成员虽然被继承(影响了派生类的内存布局和构造过程),但直接的访问途径被编译器禁止了,这是为了维护封装性。如果希望基类的某些成员能够在派生类中访问,但不暴露给更广泛的外界,应该将这些成员声明为protected。这确实是protected访问限定符的重要应用场景之一,它就是为了在继承体系内部提供访问权限,同时阻止外部直接访问。

        public继承是最常见的形式,它体现了“is-a”关系,即派生类是基类的一种。这种继承方式最符合面向对象设计的原则,支持多态、接口复用等特性,便于理解和维护代码。

        protectedprivate继承相对较少使用,它们更多地用于实现细节的封装或是为了重用基类的实现而不暴露其接口。特别是private继承,它经常被用来实现“has-a”关系,而非直接的类型层次关系。虽然这些继承方式在特定场景下有其用途,但确实不如public继承常见,且可能使代码结构变得复杂,影响代码的可读性和维护性。

因此,在设计时应谨慎考虑是否采用protectedprivate继承。

        默认继承方式

        在使用继承的时候也可以不指定继承方式,使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public

例如:

        在关键字为class的派生类当中,所继承的基类成员_name的访问方式变为private。

//基类
class Person
{
public:
	string _name = "张三"; //姓名
};

//派生类
class Student : Person //默认为private继承
{
protected:
	int _stuid;   //学号
};

        在关键字为struct的派生类当中,所继承的基类成员_name的访问方式仍为public。

//基类
class Person
{
public:
	string _name = "张三"; //姓名
};

//派生类
struct Student : Person //默认为public继承
{
protected:
	int _stuid;   //学号
};

注意: 虽然继承时可以不指定继承方式而采用默认的继承方式,但还是最好显示的写出继承方式。 

基类和派生类对象赋值转换

        派生类对象可以赋值给基类的对象、基类的指针以及基类的引用。

        但在这个过程中,会发生基类和派生类对象之间的赋值转换。

代码举例:

//基类
class Person
{
protected:
	string _name; //姓名
	string _sex;  //性别
	int _age;     //年龄
};

//派生类
class Student : public Person
{
protected:
	int _stuid;   //学号
};

        代码当中可以出现以下逻辑:

Student s;
Person p = s;     //派生类对象赋值给基类对象
Person* ptr = &s; //派生类对象赋值给基类指针
Person& ref = s;  //派生类对象赋值给基类引用

        对于这种做法,有个形象的说法叫做 切片/切割,寓意把派生类中基类那部分切来赋值过去。 

派生类对象赋值给基类对象图示:

派生类对象赋值给基类指针图示:

派生类对象赋值给基类引用图示:

注意: 基类对象不能赋值给派生类对象,基类的指针可以通过强制类型转换赋值给派生类的指针,但是此时基类的指针必须是指向派生类的对象才是安全的。 

继承中的作用域

  • 在继承体系中基类和派生类都有独立的作用域。
  • 子类和父类中有同名成员,子类成员将屏蔽父类对同名成员的直接访问,这种情况叫隐藏,也叫重定义。(在子类成员函数中,可以使用 基类::基类成员 显示访问)
  • 需要注意的是如果是成员函数的隐藏,只需要函数名相同就构成隐藏。
  • 注意在实际中在继承体系里面最好不要定义同名的成员。

代码举例:

        对于以下代码,访问成员_num时将访问到子类当中的_num。 

#include <iostream>
#include <string>

using namespace std;

//父类
class Person
{
protected:
	int _num = 111;
};

//子类
class Student : public Person
{
public:
	void fun()
	{
		cout << _num << endl;
	}
protected:
	int _num = 999;
};

int main()
{
	Student s;
	s.fun(); //999
	return 0;
}

代码结果:

        若此时我们就是要访问父类当中的_num成员,我们可以使用作用域限定符进行指定访问。

void fun()
{
	cout << Person::_num << endl; //指定访问父类当中的_num成员
}

如果是成员函数的隐藏,只需要函数名相同就构成隐藏。

代码举例:

        对于以下代码,调用成员函数fun时将直接调用子类当中的fun,若想调用父类当中的fun,则需使用作用域限定符指定类域。 

#include <iostream>
#include <string>

using namespace std;

//父类
class Person
{
public:
	void fun(int x)
	{
		cout << x << endl;
	}
};

//子类
class Student : public Person
{
public:
	void fun(double x)
	{
		cout << x << endl;
	}
};

int main()
{
	Student s;
	s.fun(3.14);       //直接调用子类当中的成员函数fun
	s.Person::fun(20); //指定调用父类当中的成员函数fun
	return 0;
}

代码结果:

        代码当中,父类中的fun和子类中的fun不是构成函数重载,因为函数重载要求两个函数在同一作用域,而此时这两个fun函数并不在同一作用域。

注意:为了避免类似问题,实际在继承体系当中最好不要定义同名的成员。 

派生类的默认成员函数

        当我们不写编译器会自动生成的函数,类当中的默认成员函数有以下六个:

        下面我们看看派生类当中的默认成员函数,与普通类的默认成员函数的不同之处。 

代码举例:

        以下面这个Person类为基类。

//基类
class Person
{
public:
	//构造函数
	Person(const string& name = "peter")
		:_name(name)
	{
		cout << "Person()" << endl;
	}
	//拷贝构造函数
	Person(const Person& p)
		:_name(p._name)
	{
		cout << "Person(const Person& p)" << endl;
	}
	//赋值运算符重载函数
	Person& operator=(const Person& p)
	{
		cout << "Person& operator=(const Person& p)" << endl;
		if (this != &p)
		{
			_name = p._name;
		}
		return *this;
	}
	//析构函数
	~Person()
	{
		cout << "~Person()" << endl;
	}
private:
	string _name; //姓名
};

        用该基类派生出Student类,Student类当中的默认成员函数的基本逻辑如下: 

//派生类
class Student : public Person
{
public:
	//构造函数
	Student(const string& name, int id)
		:Person(name) //调用基类的构造函数初始化基类的那一部分成员
		, _id(id) //初始化派生类的成员
	{
		cout << "Student()" << endl;
	}
	//拷贝构造函数
	Student(const Student& s)
		:Person(s) //调用基类的拷贝构造函数完成基类成员的拷贝构造
		, _id(s._id) //拷贝构造派生类的成员
	{
		cout << "Student(const Student& s)" << endl;
	}
	//赋值运算符重载函数
	Student& operator=(const Student& s)
	{
		cout << "Student& operator=(const Student& s)" << endl;
		if (this != &s)
		{
			Person::operator=(s); //调用基类的operator=完成基类成员的赋值
			_id = s._id; //完成派生类成员的赋值
		}
		return *this;
	}
	//析构函数
	~Student()
	{
		cout << "~Student()" << endl;
		//派生类的析构函数会在被调用完成后自动调用基类的析构函数
	}
private:
	int _id; //学号
};

派生类与普通类的默认成员函数的不同之处概括为以下几点:

  1. 构造函数:当创建一个派生类对象时,首先需要确保基类部分得到恰当的初始化。因此,派生类的构造函数执行前,会自动调用基类的构造函数。如果基类有无参构造函数(默认构造函数),编译器会自动调用它;如果没有,默认不会自动调用,这时就必须在派生类的构造函数初始化列表中显式调用一个基类的构造函数。

  2. 拷贝构造函数:当通过已有的派生类对象来创建一个新的派生类对象时,派生类的拷贝构造函数不仅要复制派生类新增的成员,还需要调用基类的拷贝构造函数来复制基类的成员,以保证新对象的基类部分与原对象相同。

  3. 赋值运算符重载:类似地,在使用赋值运算符给一个派生类对象赋值时,派生类的赋值运算符重载函数需确保不仅复制派生类特有的数据成员,还要正确地调用基类的赋值运算符来处理基类的数据成员,实现深拷贝,以维护对象状态的一致性。

  4. 析构函数:对象生命周期结束时,派生类的析构函数执行完毕后,会自动调用基类的析构函数,以确保基类资源的正确释放。这个顺序是先派生类后基类,与构造过程的顺序相反,遵循“后构造先析构”的原则,确保所有资源被妥善清理。

  5. 初始化顺序:这一点是对构造函数调用顺序的强调,明确指出在派生类对象的构造过程中,首先执行的是基类的构造函数,随后才是派生类自身的构造函数,确保从基到派的构建逻辑。

  6. 析构顺序:与初始化顺序相反,当对象生命周期结束进行清理时,派生类的析构函数先执行,完成派生类特有资源的清理,之后基类的析构函数被调用,释放基类资源,这是确保资源释放的正确顺序。

 在编写派生类的默认成员函数(如构造函数、拷贝构造函数、赋值运算符、析构函数等)时,需要注意以下几个关键点:

  1. 构造函数

    • 如果基类有非默认构造函数,且你希望在派生类构造时也调用它,你需要在派生类的构造函数初始化列表中显式调用基类的构造函数。
    • 确保初始化派生类新增的成员变量。这通常也在初始化列表中完成。
  2. 拷贝构造函数和赋值运算符

    • 如果基类定义了自己的拷贝构造函数或赋值运算符,派生类需要显式调用它们以确保基类部分被正确拷贝或赋值。这通常通过使用BaseClass(baseObj)BaseClass& operator=(const BaseClass& baseObj)的方式实现。
    • 实现派生类自己的拷贝控制成员时,要遵循“深拷贝”的原则,避免资源共享引起的问题。
  3. 析构函数

    • 派生类的析构函数不需要显式调用基类的析构函数,C++会自动保证在派生类对象销毁时,先执行派生类的析构函数,再执行基类的析构函数。但需确保在派生类析构函数中释放派生类特有的资源。
  4. 虚函数

    • 如果基类中有虚函数,确保在派生类中正确地重写这些虚函数,并考虑其行为是否符合预期。特别是当基类指针或引用指向派生类对象时,通过基类接口调用的应是派生类的实现。
  5. 访问权限

    • 注意基类成员的访问权限。派生类构造函数不能直接访问基类的私有成员,即使在初始化列表中也是如此。如果需要在派生类构造函数中操作基类私有成员,应通过基类提供的公有或受保护的接口进行。
  6. 继承方式

    • 明确指定继承方式(public, protected, private)。默认为private,可能不是你想要的。公有继承(public inheritance)通常表示“is-a”关系,确保派生类可以作为基类的替代使用。
  7. 构造顺序与析构顺序

    • 构造时,先调用基类的构造函数,再调用派生类的构造函数。析构时顺序相反,先调用派生类的析构函数,再调用基类的析构函数。

继承与友元

        友元关系不能继承,也就是说基类的友元可以访问基类的私有和保护成员,但是不能访问派生类的私有和保护成员。

代码举例:

        以下代码中Display函数是基类Person的友元,当时Display函数不是派生类Student的友元,即Display函数无法访问派生类Student当中的私有和保护成员。

#include <iostream>
#include <string>

using namespace std;

// 前向声明,为了让Display函数能够识别Student类型,这里先声明Student类
class Student;

// Person类定义
class Person
{
public:
    // 声明Display为Person类的友元函数,这意味着Display可以访问Person的所有成员,包括私有和保护成员
    friend void Display(const Person& p, const Student& s);

protected:
    string _name; // 姓名,声明为保护成员,允许派生类访问
};

// Student类,从Person公有继承
class Student : public Person
{
protected:
    int _id; // 学号,声明为保护成员
};

// Display函数实现,它是Person和Student类的友元,因此可以访问这两个类的保护和公有成员
void Display(const Person& p, const Student& s)
{
    cout << p._name << endl; // 可以访问,因为Person类将Display设为友元,_name是Person的保护成员
    cout << s._id << endl;   // 这行代码如果取消注释会编译错误,因为虽然Display是Person的友元,但并非Student的直接友元,
    //                        // 因此不能访问Student的保护成员_id。即使Student继承自Person也不行。
}

int main()
{
    Person p; // 创建一个Person对象
    Student s; // 创建一个Student对象
    Display(p, s); // 调用Display函数,传入p和s的对象
    return 0;
}

代码结果:

        若想让Display函数也能够访问派生类Student的私有和保护成员,只能在派生类Student当中进行友元声明。 

class Student : public Person
{
public:
	//声明Display是Student的友元
	friend void Display(const Person& p, const Student& s);
protected:
	int _id; //学号
};

继承与静态成员

        若基类当中定义了一个static静态成员变量,则在整个继承体系里面只有一个该静态成员。无论派生出多少个子类,都只有一个static成员实例。

代码举例:

        在基类Person当中定义了静态成员变量_count,尽管Person又继承了派生类Student和Graduate,但在整个继承体系里面只有一个该静态成员。
        我们若是在基类Person的构造函数和拷贝构造函数当中设置_count进行自增,那么我们就可以随时通过_count来获取该时刻已经实例化的Person、Student以及Graduate对象的总个数。

#include <iostream>
#include <string>

using namespace std;

// 基类 Person
class Person
{
public:
    // 默认构造函数,每当创建一个Person对象时,计数器_count加一
    Person() 
    { 
        _count++; 
    }
    // 拷贝构造函数,同样增加计数器_count,用于追踪所有Person及其派生类实例的数量
    Person(const Person& p) 
    {
        _count++;
    }
protected:
    string _name; // 姓名,保护成员,允许派生类访问
public:
    static int _count; // 静态成员变量,用于统计Person类及派生类实例的总数
};

// 在类外部对静态成员变量_count进行初始化
int Person::_count = 0;

// 派生类 Student 继承自 Person
class Student : public Person
{
protected:
    int _stuNum; // 学号,保护成员
};

// 派生类 Graduate 继承自 Person
class Graduate : public Person
{
protected:
    string _seminarCourse; // 研究科目,保护成员
};

int main()
{
    // 创建Student对象s1,由于Person类的构造函数,_count加1变为1
    Student s1;
    
    // 使用s1对象拷贝构造s2,调用Person类的拷贝构造函数,_count再次加1变为2
    Student s2(s1);
    
    // 创建Student对象s3,_count加1变为3
    Student s3;
    
    // 创建Graduate对象s4,由于Graduate也是Person的派生类,_count继续加1变为4
    Graduate s4;
    
    // 输出Person类及其所有派生类的实例总数,即_count的值,结果为4
    cout << Person::_count << endl; // 4
    
    // 注意:这里尝试输出Student类的_count是不正确的,因为静态成员是属于类的,而不是特定的派生类。
    // 所有的Person、Student和Graduate实例共享同一个_count,所以直接写Student::_count是多余的,
    // 应该只通过基类Person来访问静态成员_count。不过,由于静态成员的特性,这里实际上也会输出4,
    // 但这不代表正确的编程习惯,应该避免。
    cout << Student::_count << endl; // 4
    
    return 0;
}

        此时我们也可以通过打印Person类和Student类当中静态成员_count的地址来证明它们就是同一个变量。

cout << &Person::_count << endl;
cout << &Student::_count << endl; 

 代码结果:

继承的方式

单继承 (Single Inheritance)

  • 定义:单继承是最简单的继承形式,指一个子类(派生类)只从一个基类继承。这种继承结构清晰,易于理解。
  • 特点:简化了对象模型,减少了继承带来的复杂性,如内存布局和方法解析顺序相对简单。

多继承 (Multiple Inheritance)

  • 定义:多继承允许一个子类继承自两个或更多个基类。这使得子类能够合并多个基类的特性。
  • 特点
    • 提供了更灵活的代码复用,但也引入了潜在的复杂性,比如可能面临的“菱形问题”(Diamond Problem)。
    • 需要注意的是,多继承可能导致对象大小增加,因为子类需要包含所有基类的实例变量。

菱形继承 (Diamond Inheritance)

  • 定义:菱形继承是多继承的一种特殊情况,当一个类直接继承自两个类,而这两个类又共同继承自同一个基类时,就形成了菱形继承结构。
  • 问题:最直接的问题是如果有同名成员(尤其是数据成员)在基类中定义,那么在派生类中访问时会出现二义性。
  • 解决方案:C++通过引入虚继承(Virtual Inheritance)来解决菱形继承中的二义性问题。虚继承确保基类只被继承一次,从而在派生类中只有一个基类的副本。

        从菱形继承的模型构造就可以看出,菱形继承的继承方式存在数据冗余和二义性的问题。

代码举例:

        对于以上菱形继承的模型,当我们实例化出一个Assistant对象后,访问成员时就会出现二义性问题。

#include <iostream>
#include <string>

using namespace std;

// 定义基类Person,包含姓名成员变量
class Person
{
public:
    string _name; // 姓名
};

// 定义派生类Student,从Person公有继承,添加学号成员变量
class Student : public Person
{
protected:
    int _num; // 学号
};

// 定义派生类Teacher,从Person公有继承,添加职工编号成员变量
class Teacher : public Person
{
protected:
    int _id; // 职工编号
};

// 定义派生类Assistant,同时从Student和Teacher公有继承,形成菱形继承结构,添加主修课程成员变量
// 注意:这里没有使用虚继承,因此Person的内容被间接继承了两次
class Assistant : public Student, public Teacher
{
protected:
    string _majorCourse; // 主修课程
};

int main()
{
    Assistant a;
    
    // 尝试给Assistant对象a的_name赋值,但由于_person是通过两条路径(Student和Teacher)被继承到Assistant中,
    // 编译器无法确定应访问哪个基类版本的_name成员,导致二义性错误。
    // 解决方法是在Person类前加上virtual关键字,使Student和Teacher虚继承Person,消除二义性。
    a._name = "peter"; // 此处会导致编译错误,指出_name是二义性的

    return 0;
}

 代码结果:

        Assistant对象是多继承的Student和Teacher,而Student和Teacher当中都继承了Person,因此Student和Teacher当中都有_name成员,若是直接访问Assistant对象的_name成员会出现访问不明确的报错。 

        对于此,我们可以显示指定访问Assistant哪个父类的_name成员。

//显示指定访问哪个父类的成员
a.Student::_name = "张同学";
a.Teacher::_name = "张老师";

注意:虽然该方法可以解决二义性的问题,但仍然不能解决数据冗余的问题。因为在Assistant的对象在Person成员始终会存在两份。 

 

菱形虚拟继承

        为了解决菱形继承的二义性和数据冗余问题,出现了虚拟继承。

        如前面说到的菱形继承关系,在Student和Teacher继承Person是使用虚拟继承,即可解决问题。

代码如下:

#include <iostream>
#include <string>

using namespace std;

// 基类 Person,定义姓名成员变量
class Person
{
public:
    string _name; // 姓名
};

// 派生类 Student,使用虚拟继承自 Person,添加学号成员变量
class Student : virtual public Person
{
protected:
    int _num; // 学号
};

// 派生类 Teacher,使用虚拟继承自 Person,添加职工编号成员变量
class Teacher : virtual public Person
{
protected:
    int _id; // 职工编号
};

// 派生类 Assistant,同时继承自 Student 和 Teacher,由于 Student 和 Teacher 都是虚拟继承自 Person,
// 因此 Person 的内容在 Assistant 中只有一份实例,消除了二义性问题。
class Assistant : public Student, public Teacher
{
protected:
    string _majorCourse; // 主修课程
};

int main()
{
    Assistant a;
    
    // 现在可以为 a 对象的 _name 成员赋值,因为通过虚拟继承消除了二义性。
    a._name = "peter"; // 无二义性,正确设置姓名

    return 0;
}

        此时就可以直接访问Assistant对象的_name成员了,并且之后就算我们指定访问Assistant的Student父类和Teacher父类的_name成员,访问到的都是同一个结果,解决了二义性的问题。 

cout << a.Student::_name << endl; 
cout << a.Teacher::_name << endl; 

        打印Assistant的Student父类和Teacher父类的_name成员的地址时,显示的也是同一个地址,解决了数据冗余的问题。

cout << &a.Student::_name << endl; 
cout << &a.Teacher::_name << endl; 

代码结果:

菱形虚拟继承原理

        若不使用菱形虚拟继承时,以下菱形继承当中D类对象的各个成员在内存当中的分布情况。

 

代码如下:

#include <iostream>

using namespace std;

class A
{
public:
	int _a;
};

class B : public A
{
public:
	int _b;
};

class C : 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;
}

        通过内存窗口,我们可以看到D类对象当中各个成员在内存当中的分布情况如下: 

 

        也就是说,D类对象当中各个成员在内存当中的分布情况如下:

 

        可以看出为什么菱形继承导致了数据冗余和二义性,根本原因就是D类对象当中含有两个_a成员。 

        使用菱形虚拟继承时,以下菱形继承当中D类对象的各个成员在内存当中的分布情况。

 

代码如下:

#include <iostream>

using namespace std;

class A
{
public:
	int _a;
};

class B : virtual public A
{
public:
	int _b;
};

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;
}

        通过内存窗口,我们可以看到D类对象当中各个成员在内存当中的分布情况如下: 

        其中D类对象当中的_a成员被放到了最后,而在原来存放两个_a成员的位置变成了两个指针,这两个指针叫虚基表指针,它们分别指向一个虚基表。
        虚基表中包含两个数据,第一个数据是为多态的虚表预留的存偏移量的位置(这里我们不必关心),第二个数据就是当前类对象位置距离公共虚基类的偏移量。
        也就是说,这两个指针经过一系列的计算,最终都可以找到成员_a。

 

        若是将D类对象赋值给B类对象,在这个切片过程中,就需要通过虚基表中的第二个数据找到公共虚基类A的成员,得到切片后该B类对象在内存中仍然保持这种分布情况。

D d;
B b = d; //切片行为

        得到切片后该B类对象当中各个成员在内存当中的分布情况如下: 

 

        其中,_a对象仍然存储在该B类对象的最后。 

继承的总结和反思

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

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

        3. 继承和组合

  • public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。
  • 组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。
  • 优先使用对象组合,而不是类继承 。
  • 继承允许你根据基类的实现来定义派生类的实现。这种通过生成派生类的复用通常被称为白箱复用(white-box reuse)。术语“白箱”是相对可视性而言:在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高。
  • 对象组合是类继承之外的另一种复用选择。新的更复杂的功能可以通过组装或组合对象来获得。对象组合要求被组合的对象具有良好定义的接口。这种复用风格被称为黑箱复用(black-box reuse),因为对象的内部细节是不可见的。对象只以“黑箱”的形式出现。组合类之间没有很强的依赖关系,耦合度低。优先使用对象组合有助于你保持每个类被封装。
  • 实际尽量多去用组合。组合的耦合度低,代码维护性好。不过继承也有用武之地的,有些关系就适合继承那就用继承,另外要实现多态,也必须要继承。类之间的关系可以用继承,可以用组合,就用组合。

继承关系(is-a)

class Car 
{
protected:
    string _colour = "白色"; // 颜色
    string _num = " "; // 车牌号
};

class BMW : public Car 
{
public:
    void Drive() {cout << "好开-操控" << endl;}
};

class Benz : public Car 
{
public:
    void Drive() {cout << "好坐-舒适" << endl;}
};
  • Car类定义了一些基本属性,如颜色和车牌号。
  • BMWBenz类通过公有继承(public)自Car类,表示宝马车和奔驰车都是车的一种,它们“是”车。这种继承关系体现了“is-a”原则,宝马和奔驰车具备车的基本属性(颜色、车牌号),同时各自还定义了特有的Drive()方法来展示不同品牌的驾驶体验。

组合关系(has-a)

class Tire 
{
protected:
    string _brand = "Michelin";  // 品牌
    size_t _size = 17;           // 尺寸
};

class Car 
{
protected:
    string _colour = "白色";     // 颜色
    string _num = " ";           // 车牌号
    Tire _t;                     // 轮胎
};
  • Tire类定义了一个轮胎应有的属性,如品牌和尺寸。
  • 在重新定义的Car类中,通过包含一个Tire类型的成员变量_t,表明一辆车“有”一个轮胎。这种包含关系体现了“has-a”原则,即车并不继承轮胎的行为或状态,而是直接拥有轮胎作为其组成部分。

注意:若是两个类之间既可以看作is-a的关系,又可以看作has-a的关系,则优先使用组合。 

相关笔试面试题

什么是菱形继承?菱形继承的问题是什么?

        菱形继承是多继承的一种特殊情况,两个子类继承同一个父类,而又有子类同时继承这两个子类,我们称这种继承为菱形继承。
        菱形继承因为子类对象当中会有两份父类的成员,因此会导致数据冗余和二义性的问题。

什么是菱形虚拟继承?如何解决数据冗余和二义性?

 

        菱形虚拟继承是指在菱形继承的腰部使用虚拟继承(virtual)的继承方式,菱形虚拟继承对于D类对象当中重复的A类成员只存储一份,然后采用虚基表指针和虚基表使得D类对象当中继承的B类和C类可以找到自己继承的A类成员,从而解决了数据冗余和二义性的问题。

继承和组合的区别?什么时候用继承?什么时候用组合?

        继承是一种is-a的关系,而组合是一种has-a的关系。如果两个类之间是is-a的关系,使用继承;如果两个类之间是has-a的关系,则使用组合;如果两个类之间的关系既可以看作is-a的关系,又可以看作has-a的关系,则优先使用组合。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/659162.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

「盘点」JetBrains IDEs v2024.1新功能一览,更智能的开发体验!

JetBrains IDEs日前正式发布了v2024.1版本&#xff0c;此版本中最大的亮点就是带来了AI赋能的全行代码补全&#xff0c;同时在最新的IDEs中重做了终端、拥有更强大的代码编辑和导航功能、更智能的代码分析和提示、更优化的性能、更丰富的插件和集成等。总的来说&#xff0c;Jet…

淘宝API探秘:一键获取店铺所有商品的魔法之旅

在数字时代的今天&#xff0c;数据已经成为了商业世界中的魔法石。而对于淘宝店主或者那些想要深入探索淘宝数据的人来说&#xff0c;淘宝API就像是打开阿里巴巴宝藏库的钥匙。今天&#xff0c;我们就来一起探索如何使用淘宝API&#xff0c;特别是如何获取店铺所有商品的接口&a…

为WPF的Grid添加网格边框线

在WPF中使用Grid绘制表格的时候&#xff0c;如果元素较多、排列复杂的话&#xff0c;界面会看起来很糟糕&#xff0c;没有层次&#xff0c;这时用网格或边框线分割各元素&#xff08;标签或单元格&#xff09;将会是页面看起来整齐有条理。 默认没有边框线的如下图所示&#xf…

Java 循环嵌套深度揭秘:挑战极限与性能优化

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 探索Java的调用栈极限 在Java中&#xff0c;方法调用是通过栈&#xff08;Stack&#xff09;这种数据结构来实现的。每当一个方法被调用时&#xff0c;一个新的栈帧&#xff08;Stack Frame&#xff09;会被创建并压…

React 中的 Fiber 架构

React Fiber 介绍 React Fiber 是 React 的一种重写和改进的核心算法&#xff0c;用于实现更细粒度的更新和高效的调度。它是 React 16 版本中的一个重要更新&#xff0c;使得 React 能够更好地处理复杂和高频的用户交互。以下是对 React Fiber 的详细介绍&#xff1a; 为什么…

便民社区信息小程序源码系统 功能强大 带生活电商+求职招聘功能 带完整的安装代码包以及搭建教程

系统概述 便民社区信息小程序源码系统是一款集多种功能于一身的综合性平台。它旨在为用户提供便捷的生活服务&#xff0c;满足社区居民的各种需求。无论是购物、求职还是获取社区信息&#xff0c;都能在这个平台上得到满足。该系统采用先进的技术架构&#xff0c;确保系统的稳…

【python 进阶】 绘图

1. 将多个柱状绘制在一个图中 import seaborn as sns import matplotlib.pyplot as plt import numpy as np import pandas as pd# 创建示例数据 categories [A, B, C, D, E] values1 np.random.randint(1, 10, sizelen(categories)) values2 np.random.randint(1, 10, siz…

揭秘!编写高质量代码的关键:码农必知的黄金法则!

文章目录 一、保持代码的简洁与清晰二、遵循良好的命名规范三、注重代码的可读性四、利用抽象与封装五、遵循SOLID原则六、关注代码性能七、确保代码安全性《码农修行&#xff1a;编写优雅代码的32条法则》编辑推荐内容简介目录前言/序言 在编程的世界里&#xff0c;每一位码农…

VSCode 报错 之 运行 js 文件报错 ReferenceError: document is not defined

1. 背景 持续学习ing 2. 遇到的问题 在VSCode 右键 code runner js 文件报错 ReferenceError: document is not defined eg&#xff1a; // 为每个按钮添加点击事件监听器 document.querySelectorAll(button).forEach(function (button) {button.addEventListener(click, f…

python基础-数据结构-leetcode刷题必看-heapq --- 堆队列算法

文章目录 堆的定义堆的主要操作堆的构建堆排序heapq模块heapq.heappush(heap, item)heapq.heappop(heap)heapq.heappushpop(heap, item)heapq.heapreplace(heap, item)heapq.merge(*iterables, keyNone, reverseFalse)heapq.nlargest(n, iterable, keyNone)heapq.nsmallest(n, …

赛氪网与武汉外语外事职业学院签署校企合作,共创职业教育新篇章

5月23日下午14:00&#xff0c;武汉外语外事职业学院在藏龙岛校区食堂三楼报告厅隆重举行了2024年职业教育活动周优秀校外实习基地表彰仪式。本次活动旨在表彰在职业教育领域作出突出贡献的校外实习基地&#xff0c;同时加强校企合作&#xff0c;共同推动职业教育的发展。作为重…

gitlab之docker-compose汉化离线安装

目录 概述离线资源docker-compose结束 概述 gitlab可以去 hub 上拉取最新版本&#xff0c;在此我选择汉化 gitlab &#xff0c;版本 11.x 离线资源 想自制离线安装镜像&#xff0c;请稳步参考 docker镜像的导入导出 &#xff0c;无兴趣的直接使用在此提供离线资源 百度网盘(链…

经典文献阅读之--RepViT-SAM(利用语义分割提高NDT地图压缩和描述能力的框架)

0. 简介 Segment Anything Model (SAM) 最近在各种计算机视觉任务上展现了令人瞩目的零样本迁移性能 。然而&#xff0c;其高昂的计算成本对于实际应用仍然具有挑战性。MobileSAM 提出通过使用蒸馏替换 SAM 中的重图像编码器&#xff0c;使用 TinyViT&#xff0c;从而显著降低了…

认识K8s集群的声明式资源管理方法

前言 Kubernetes 集群的声明式资源管理方法是当今云原生领域中的核心概念之一&#xff0c;使得容器化应用程序的部署和管理变得更加高效和可靠。本文将认识了解 Kubernetes 中声明式管理的相关理念、实际应用以及优势。 目录 一、管理方法介绍 1. 概述 2. 语法格式 2.1 管…

AI图书推荐:用ChatGPT和Python搭建AI应用来变现

《用ChatGPT和Python搭建AI应用来变现》&#xff08;Building AI Applications with ChatGPT API&#xff09;将ChatGPT API与Python结合使用&#xff0c;可以开启构建非凡AI应用的大门。通过利用这些API&#xff0c;你可以专注于应用逻辑和用户体验&#xff0c;而ChatGPT强大的…

适合学生党的蓝牙耳机有哪些?盘点四大性价比蓝牙耳机品牌

对于追求高品质音乐体验而又预算有限的学生党来说&#xff0c;一款性价比高的蓝牙耳机无疑是最佳选择&#xff0c;在众多品牌和型号中&#xff0c;如何挑选到既适合自己需求又价格亲民的蓝牙耳机&#xff0c;确实是一个值得思考的问题&#xff0c;作为一个蓝牙耳机大户&#xf…

台灯护眼是真的吗?警惕这六大问题!

在当今社会&#xff0c;随着电子设备的普及和长时间的用眼&#xff0c;大多数人面临着严重的视觉疲劳问题。长时间盯着屏幕或学习&#xff0c;眼睛需要不断调节焦距&#xff0c;导致眼睛肌肉疲劳&#xff0c;进而引发视力下降。这种现象在年轻一代甚至青少年中尤为普遍&#xf…

半导体测试基础 - 功能测试

功能测试(Functional Test)主要是验证逻辑功能,是运用测试矢量和测试命令来进行的一种测试,相比于纯 DC 测试而言,组合步骤相对复杂且耦合度高。 在功能测试阶段时,测试系统会以周期为单位,将测试矢量输入 DUT,提供预测的结果并与输出的数据相比较,如果实际的结果与测…

图论(五)-最短路

一、Bellman-Ford算法 算法思想&#xff1a;通过 n 次循环&#xff0c;每次循环都遍历每条边&#xff08;共 m 条边&#xff09;&#xff0c;进而更新节点的距离&#xff0c;每次循环至少可以确定一个点的最短路&#xff0c;循环 n 次&#xff0c;求出 n 个点的最短路 时间复杂…

opencascade V3d_RectangularGrid 源码学习

类V3d_RectangularGrid V3d_RectangularGrid() V3d_RectangularGrid::V3d_RectangularGrid(const V3d_ViewerPointer &aViewer, const Quantity_Color &aColor, const Quantity_Color &aTenthColor) // 构造函数 ◆ ~V3d_RectangularGrid() virtual V3d_Rectang…