【C++】C++中的继承

news2025/1/23 12:20:08

目录

  • 一.继承的概念和定义
    • 1.继承的概念
    • 2.继承定义
      • 2.1定义格式
      • 2.2继承关系和访问限定符
      • 2.3继承基类成员访问方式的变化
  • 二.基类和派生类对象赋值转换
  • 三.继承中的作用域
  • 四.派生类的默认成员函数
  • 五.继承和友元
  • 六.继承与静态成员
  • 七.复杂的菱形继承及菱形虚拟继承
    • 1.单继承
    • 2.多继承
    • 3.菱形继承
      • 3.1菱形继承的问题
      • 3.2菱形虚拟继承
        • (1)解决二义性
        • (2)解决冗余
      • 3.3笔试题
  • 八.组合
    • 继承与组合区别:
  • 九.继承的总结和反思
  • 十.经典面试题

一.继承的概念和定义

1.继承的概念

继承 机制是面向对象程序设计使代码可以复用 的最重要的手段,它允许程序原在保持原有类特性的基础上进行扩展 ,增加功能,这样产生新的类,称为派生类 。继承 呈现了面向对象程序设计的层次结构 ,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用

比方说我们现在定义一个Person类,它主要的属性如下,当我们在定义一个Student类时,就可以直接继承Person类,利用Person类中已有的属性和方法,减少代码的冗余:

在这里插入图片描述

编写如下代码进行测试查看其测试结果:

#include<iostream>
#include<string>
#include<vector>
using namespace std;

typedef long long LL;

class Person {
public:
	Person()
	{
		cout << "Person()" << endl;
	}
	~Person()
	{
		cout << "~Person()" << endl;
	}
protected:
	string _name = "张三";      //姓名
	LL _phone = 123456798;     //电话
	int _age = 18;             //年龄
	int _height = 25;          //身高
};

class Student:public Person {
public:
	void Print()
	{
		cout << _name << " " << _phone << " " << _age << " " << _height << endl;
	}
private:
	int _score;   //成绩
};

int main()
{
	Student s;
	s.Print();
	cout << "sizeof(Person):" << sizeof(Person) << endl;
	cout << "sizeof(s):" << sizeof(s) << endl;

	return 0;
}

在这里插入图片描述

观察上面的代码和运行结果,我们可以发现,子类继承了父类之后可以在子类中使用父类的成员(可否还有继承关系和访问限定符的约束下面会讲)。

观察Person类的大小和Student创建对象后对象的大小,我们可以得知Person类也就是父类中的成员会成为子类的一部分。

总结:

  • 子类继承父类后,父类的成员会变为子类的一部分(而子类可不可以直接使用这些成员与继承关系,和这些成员的访问限定符有关)

2.继承定义

2.1定义格式

下面我们看到Person是父类,也称为基类。Student是子类,也称作派生类。(基类和派生类会在本文中多次提到)

在这里插入图片描述

2.2继承关系和访问限定符

有关于继承关系的修饰符,于我们所熟知的面向对象的三种限定类成员的限定符相同,有以下三种组成:

在这里插入图片描述

而使用不同的继承方式继承父类,在面对父类中不同的访问限定符修饰的成员,会产生不同的结果,如下:

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

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

以横轴为protected继承 ,纵轴为基类的protected成员 为例,表示使用protected符号继承,并且父类中使用protected修饰的成员相当于在子类中使用protected修饰的成员一样。

编写如下代码,演示不同访问限定符下,使用不同的继承关系继承的结果:

class Person {
public:
	void Pub()
	{
		cout << "public" << endl;
	}
protected:
	void Pro()
	{
		cout << "protected" << endl;
	}
private:
	void Pri()
	{
		cout << "private" << endl;
	}
};
//class Student : private Person
//class Student : protected Person
class Student : public Person
{
public:
	void Print()
	{
		Pub();
		Pro();
		//Pri();//private成员,调用就会报错
	}
};

在这里插入图片描述

总结:

  1. 基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指基类的私有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是在类外面都不能取访问它。
  2. 基类private成员在派生类中是不能被访问,如果基类成员不想再类外直接被访问,但需要再派生类中能访问,就定义为protected。可以看出保护成员限定符是因继承才出现的。
  3. 实际上面的表格我们进行一下总结会发现,基类的私有成员再子类都是不可见,基类的其他成员在子类的访问方式 == Min(成员在基类的访问限定符,继承方式),public > protected > private
    • 从继承的概念中我们便得知了子类继承父类,父类的成员会变为子类的一部分 ,所以使用provate继承,基类的private成员也继承到了子类中,只是不能使用。
  4. 继承方式也可以不写,编译器会给一个默认的继承方式,使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过最好显示的写出继承方式。
  5. 在实际运用中一般使用都是public继承,几乎很少使用protect/private继承 ,也不提倡使用protected/private继承,因为这样继承下来的成员都只能在派生类的类里面使用,实际中扩展维护性不强。

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

  • 派生类的对象 可以赋值给基类的对象/基类的指针/基类的引用 。这里有个形象的说法叫切片或切割。寓意把派生类中父类的那部分切来赋值过去。

    在这里插入图片描述

    编写如下测试代码:

    class Person {
    protected:
    	Person()
    	{
    		cout << "Person" << endl;
    	}
    private:
    	int _age;
    };
    
    class Student : public Person{
    private:
    	int Stu_id;
    };
    
    int main()
    {
    	Student s;
    	Person p = s;
    	Person& rp = s;
    	Person* ptrp = &s;
    
    	return 0;
    }
    

    该代码能正常运行,其结果如下:

    在这里插入图片描述

    只显示了一个Person表示只调用了一次Person类构造函数,即Student类对象创建时调用(关于构造函数下面会讲)

    所以派生类的对象 赋值给基类的对象/基类的指针/基类的引用 根据切片 可以看作是进行了如下操作:

    在这里插入图片描述

    测试1:基类的指针/基类的引用指向子类包含的父类成员

    我们在测试子类对象赋值给父类的引用,通过引用修改父类的成员变量的值,在查看子类对象对应的值是否改变来测试上图是否正确:

    class Person {
    public://访问限定符改为public,方便引用访问成员函数
    	Person()
    	{
    		cout << "Person" << endl;
    	}
    	void change_age(int age)
    	{
    		_age = age;
    	}
    	void Print_age()
    	{
    		cout << "_age:" << _age << endl;
    	}
    private:
    	int _age = 1;
    };
    
    class Student : public Person{
    private:
    	int Stu_id;
    };
    
    int main()
    {
    	Student s;
    	Person& rp = s;
    	rp.change_age(4);//修改父类的成员变量
    	s.Print_age();//查看父类的成员变量是否修改
    
    	return 0;
    }
    

    在这里插入图片描述

    子类对应的父类的成员变量改变,证明引用只是引用子类包含的父类的成员,同时也能证明父类指针类型指向的是子类包含的父类的成员(这里的道理是相通的)。

    测试2:没有临时变量

    对于下面的代码,变量b引用变量a时,因为类型的不同会进行隐式类型转换,此时会产生一个临时变量,来存放变量a类型转换后的值,而临时变量具有常性,需要增加const 修饰。

    	double a = 1.1;
    	const int& b = a;
    

    而对于我们上面的测试代码中,同样时引用,同样是不同的类型,而没有使用const 修饰,证明没有临时变量的产生。

    测试3:派生类的对象赋值给基类的对象,调用父类的拷贝构造,在下面的派生类的默认成员函数中会讲。

  • 基类对象不能赋值给派生类对象。

    子可以给父,因为子继承了父亲的成员,而当父类要赋值给子类时,子类包含父类的成员和自己的成员,而父类只有自己的成员,赋值注定是不对等的。

  • 基类的指针或者引用可以通过强制类型转换赋值给派生类的指针或者引用。但是必须是基类的指针是指向派生类对象时才安全的。这里基类如果是多态类型,可以使用RTTI(Run-Time Type Information)的dynamic_cast来进行识别后进行安全转换。

注意:

​ 父类接收子类这种情况,只要在继承关系为public 的时候才会有效,因为只有继承关系是public那保存在子类中的父类成员的权限都不会发生改变,而当继承关系是其它类型比如说protected ,此时存在子类中原本public限定符访问权限会改为protected,原始的权限发生改变,在经过赋值就是将protected的权限改为public这是不被允许的。

​ 之所以不被允许,是因为当你在创建一个子类时只需要使用protected 继承,只在类内修改和使用父类的成员,而经过赋值之后,获得子类对应的父类成员的变量就可以肆无忌惮的访问和修改对应的父类的成员,这与最初的编写代码时的目的背道而驰。

三.继承中的作用域

  1. 在继承体系中基类派生类 都有独立的作用域

  2. 子类和父类中允许有同名成员(作用域不同),子类成员将屏蔽父类对同名成员的直接访问,这种情况叫隐藏,也叫重定义。 (在子类成员函数中,可以使用基类::基类成员 显示访问 )

    class Person {
    protected:
    	int _age = 18;
    	int _identity = 123456;
    };
    
    class Student:public Person {
    public:
    	void Print()
    	{
    		cout << "Student-_age:" << _age << endl;        //子类中的成员变量
    		cout << "Person-_age:" << Person::_age << endl; //父类中的成员变量
    	}
    private:
    	int _age = 20;
    };
    
    int main()
    {
    	Student s;
    	s.Print();
    
    	return 0;
    }
    

    在这里插入图片描述

  3. 需要注意的是如果是成员函数的隐藏,只需要函数名相同就构成隐藏。(不需要考虑参数,也不需要考虑返回值)

    class Father {
    public:
    	void Function()
    	{
    		cout << "Father Function" << endl;
    	}
    };
    
    class Child:public Father {
    public:
    	void Function(int a)
    	{
    		cout << "Child Function " << a << endl;
    	}
    };
    
    int main()
    {
    	Child c;
    	c.Function(10);       //同名函数构成隐藏,调用子类的函数
    	c.Father::Function(); //声明调用父类的函数
        //错误写法,缺少参数,同名函数以及构成隐藏,只有子类的该函数可以直接,父类的需要声明
        //c.Function();   
        
    	return 0;
    }
    

    在这里插入图片描述

  4. 注意在实际中,继承体系 最好不要定义同名成员

    定义同名成员构成隐藏会存在隐患,最好不要给自己找麻烦

四.派生类的默认成员函数

6个默认成员函数,“默认”的意思就是指我们不写,编译器会帮我们自动生成一个,那么派生类中,这几个成员函数是如何生成的呢?

  1. 派生类的构造函数必须调用基类的构造函数初始化基类的那一部分成员。如果基类没有默认的构造函数,则必须在派生类构造函数的初始化列表阶段显示调用(即调用有参的构造函数)。
  2. 派生类的拷贝构造函数必须调用基类的拷贝构造完成基类的拷贝初始化。
  3. 派生类的operator=必须要调用基类的operator=完成基类的复制。
  4. 派生类的析构函数会在被调用完成后自动调用基类的析构函数清理基类成员。因为这样才能保证派生类对象先清理派生类成员再清理基类成员的顺序。
    • 一方面,对象的创建过程是一个压栈的过程,创建是先调用父类的析构函数,在调用子类的析构函数,所以析构出栈时,是先出子类,在出父类。
    • 其次,为了安全考虑,子类成员和父类成员的关联可能会很密切,若是先析构父类,在析构子类,可能会导致子类中某些成员的指向为野指针,造成程序崩溃。
    • 最后,由于多态的原因,在子类的析构函数中,我们无法直接调用父类的析构函数,其中父类和子类的析构函数都会被处理成destructor 造成重名,我们只能使用基类::基类析构函数 的方式调用,并且在我们自己手动调用后,编译器仍会再次调用父类的析构函数,所以析构函数我们不要取显示调用。
  5. 派生类对象初始化先调用基类构造再调基类的析构。
  6. 因为后续一些场景析构函数需要构成重写,重写的条件之一是函数名相同(这个我们后面会讲解)。那么编译器会对析构函数名进行特殊处理,处理成destructor(),所以父类析构函数不加virtual的情况下,子类析构函数和父类析构函数构成隐藏关系。

编写如下测试代码:

class Father {
public:
	Father(const int age = 10)
		:_age(age)
	{
		cout << "Father()" << endl;
	}
	Father(const Father& f)
	{
		cout << "Father(const Father& f)" << endl;
	}
	Father& operator=(const Father& f)
	{
		cout << "Father& operator=(const Father& f)" << endl;
		return *this;
	}
	~Father()//父类析构
	{
		cout << "~Father()" << endl;
	}
protected:
	int _age;
};

class Child:public Father {
public:
	Child(const int age = 10,const int num = 20)
		:Father(age)             //调用父类的构造函数
		,_num(num)
	{
		cout << "Child()" << endl;
	}
	Child(const Child& c)
		:Father(c)               //调用父类的拷贝构造
		,_num(c._num)
	{
		cout << "Child(const Child& c)" << endl;
	}
	Child& operator=(const Child& c)
	{
		if (this != &c)
		{
			Father::operator=(c);//调用父类的赋值重载
			_num = c._num;
		}
		cout << "Child& operator=(const Child& c)" << endl;
		return *this;
	}
	~Child()//子类析构
	{
		cout << "~Child()" << endl;
	}
private:
	int _num;
};

int main()
{
	Child c1;
	Father f = c1;
	Child c2 = c1;
	c2 = c1;

	return 0;
}

在这里插入图片描述

  • 其中对象释放的顺序,是按照对象的创建顺序,先创建的后释放,后创建的先释放,这是一个出栈操作。
  • 在调用拷贝构造和赋值重载时,子类中将子类对象作为参数传递给父类,由父类类型的参数接收,利用的正式上面我们所学到的切片。

总结:

​ 虽然继承中,子类包含了父类的成员,但在进行赋值拷贝等操作时,仍然需要父类的相关构造函数的帮助,子类无法对父类的成员进行这方面的操作。

​ 简单的说,就是子类完成子类的工作,父类完成父类的工作。

​ 这一块的知识我们不要局限与子类包含了父类的成员,毕竟在初始化调用构造函数和析构时,都需要父类的参与,那其它情况下,也必然需要父类来完成它所属的成员的对应操作。

五.继承和友元

友元关系不能继承 ,也就是说基类友元不能访问子类私有和保护成员。

在这里插入图片描述

在这里插入图片描述

想要让友元函数或是友元类也能访问子类,只能子类中添加对应的友元关系:

在这里插入图片描述

六.继承与静态成员

基类定义了static静态成员,则整个继承体系里面只有一个这样的成员。无论派生出多少个子类,都只有一个static成员实例。

编写如下测试代码查看,父子类是否共用一个静态成员,查看它们的静态成员地址是否相同

class Father {
public:
	int age = 10;
	static int count;
};

int Father::count = 0;

class Child :public Father{
private:
	string _name = "张三";
};

int main()
{
	Father f;
	Child c;
	cout << &(f.age) << endl;
	cout << &(c.age) << endl;

	cout << &(f.count) << endl;
	cout << &(c.count) << endl;

	return 0;
}

在这里插入图片描述

小问题:

  1. 如何创建一个无法被继承的类?

    答:使用private限定符修饰其构造函数

  2. 使用private修饰构造函数那我们也无法使用这个类创建对象,该如何解决?

    答:在类内public修饰的区域创建一个static修饰的函数CreateObject ,返回值为该类的类型,函数体内返回该类的匿名对象,即在类内调用构造函数,我们可以在类外根据类名调用这个函数创建对象。

  3. 子类能否利用这个函数完成调用构造函数?

    答:不能,子类在创建对象时需要调用父类的构造函数,而不是创建出一个父类的对象。

代码如下:

class A {
public:
	static A Create_object(const int age = 10)
	{
		return A(age);
	}
	void Print()
	{
		cout << _age << endl;
	}

private:
	A(int age = 10)
		:_age(age)
	{}

	int _age;
};

int main()
{
	A b = A::Create_object(20);
	b.Print();

	return 0;
}

在这里插入图片描述

七.复杂的菱形继承及菱形虚拟继承

事实上,上面的内容掌握后,C++继承的内容已经基本搞定,但是C++语言在创建时却为继承留下了一个大坑,需要我们明白它的问题所在即解决方法,如下:

C++的继承是即允许单继承也允许多继承,单继承就是一个子类有一个父类,多继承则是一个子类有多个父类。

1.单继承

一个子类只有一个直接父类时这个继承关系称之为单继承

在这里插入图片描述

2.多继承

一个子类有两个或两个以上的直接父类时称这个继承关系为多继承。

在这里插入图片描述

3.菱形继承

一个子类是多继承关系,而它的父类们却继承了同一个或多个类,这样的关系称之为菱形继承。

在这里插入图片描述

3.1菱形继承的问题

观察下面的对象成员模型:

在这里插入图片描述

菱形继承有数据冗余和二义性的问题,在Child的对象中Person成员会有两份。

class Person {
public:
    Person(){}
	Person(string name, int age)
		:_name(name)
		,_age(age)
	{}
protected:
	string _name;
	int _age;
};

class Father : public Person{
public:
    Father(){}
	Father(string name, int age, int weight)
		:Person(name, age)
		, _weight(weight)
	{}
protected:
	int _weight;
};

class Mather : public Person {
public:
    Mather(){}
	Mather(string name, int age, int height)
		:Person(name,age)
		,_height(height)
	{}
protected:
	int _height;
};

class Child :public Father,public Mather{
public:
    Child(){}
	Child(string name,int age,int weight,int height,string feature)
		:Father(name,age,weight)
		,Mather(name,age,height)
		,_feature(feature)
	{}
protected:
	string _feature;
};

其中,二义性我们可以为其赋予相同的值来勉强解决它,若是使用其它方法为其赋予不同的值,在访问时会编译器会报错。

int main()
{
	Child c("张三",18,120,180,"篮球");

	return 0;
}

而数据冗余,也就是空间浪费无法使用常规的方法解决,这里我们要使用virtual 关键字,修饰最底层子类的父类,也就是中间层(这里只说结果,具体的细节在下面):

在这里插入图片描述

将Child两个父类的声明修改如下:

class Father : virtual public Person{}
class Mather : virtual public Person{}

观察使用virtual 关键字前后,Child类的大小变化:

int main()
{
	cout << "sizeof(Child):" << sizeof(Child) << endl;
    return 0;
}

在这里插入图片描述

使用virtual后,Child类大小有了明显的变化。

3.2菱形虚拟继承

像上面子类继承添加virtual 关键字的父类我们称之为虚拟继承 ,虚拟继承可以同时解决菱形继承的冗余和二义性的问题.

下面我们在来看一下虚拟继承解决这两个问题的原理:

首先,我们先写一个简化的菱形继承体系,如下:

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;

	return 0;
}

(1)解决二义性

运行代码查看不使用虚拟继承的时候,D类中个成员在内存中的情况:

在这里插入图片描述

很明显,d对象中的_a 存在二义性,有两个值。

接着,我们为B、D类 添加virtual 修饰,运行代码,再来查看D类中个成员在内存中的情况:

在这里插入图片描述

  1. 原本存放B、C类所对应的_a成员的值的空间都变为了一个地址(我使用的编译器为VS2019,X86下,存储方式为小端存储)。

  2. 这两个地址所指向的空间中,第一块所占四个字节的空间我们不去讨论和本文关系不大,第二个空间存放的是一个数字。

比如:0x00CFFD3C (d对象的第一块空间)存放的的空间地址为0x001d9be0 ,所以第一块四个字节的空间地址为0x001d9be0 ,内容为0 (16进制),第二块为0x001d9be4 ,值为14 (16进制)

  1. 该数字表示目前距离公共父类第一个成员的偏移量

比如:0x00CFFD3C 的偏移量为0x001d9be4 的值14(16进制),增加偏移量后所代表的地址为0x00CFFD50 ,空间内存放的值为2

在这里插入图片描述

  1. 最终,菱形继承二义性的问题得以解决,公共父类的成员只有一个值。

注意: 上述代码中B类和C类有共同的父类A类,所以存放偏移量的地址B和C类各有一个,在菱形继承中,这种地址的存在与否只和是否继承共同父类有关 ,若是出现更复杂的情况,那存放偏移量的地址就会变得更复杂,如下图:

在这里插入图片描述

F、G有指向相同位置的偏移量存储地址,B、C有指向相同位置的偏移量存储地址,C、E有指向相同位置的偏移量存储地址

相信大家看到这里一定有很多问题,比如说为什么_a 的值为2,而不是1等等,下面我们在虚继承的状态下,调试运行一下上述的代码,相信很多问题,都会迎刃而解。

在这里插入图片描述

问题:

  1. B、C类中存储偏移量的两个四字节的空间是则么创建的?

    如上图,在对象d创建后,编译器根据菱形继承中B、C类继承了A类的情况,为它们分别分配了两个空间,用来存放指向A类第一个成员偏移量。

  2. 为什么最后A类成员_a的值为2?

    观察上图,在运行完30行代码后,A类成员_a的值为1,运行完31行后,A类成员_a的值变为2,公共父类的成员在菱形继承中不在存在冗余,只有一份,那它和普通成员无异,以最后一次修改的为主 ,所以最后_a的值为2,而不是1.

(2)解决冗余

这里为了方便解说,我们按照上面的代码将A类定义为初始父类,B,C类虚拟继承A类,D类继承B、C类。

在这里插入图片描述

  1. 虚拟继承存放偏移量的空间是固定的,不论创建出多少D类对象,所存放偏移量的空间是相同的。

    如上图,所有D对象存放偏移量的空间都为0x1d9be00x001d9be8 ,所以当对象足够多时这种独立出去的空间完全可以忽略不记。

  2. 当D类对象d1所用的父类A或B的成员空间大,D类对象d2所用的父类A或B空间少时,偏移量也是固定不变的。

    假设d1所用成员_c消耗50字节空间,d2所用成员_c消耗4字节空间,B、C的偏移量还是相同的。编译器会将_c中多余的空间在其它地方开辟一个空间出来存放数据,而在原始的位置上存放指向新开辟的空间的地址。

    我们编写如下测试代码:

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

    其中对象d比对象d1占用的空间一定要大,我们通过调试来查看它们的内存情况,如下图:

    在这里插入图片描述

    注意:

    1. (以d对象为例)d对象和d1对象中B类所属的第二块空间中,d对象这块空间的地址为0x0113FCDC,其中的数据为0x0001550e07,为一块地址,所指向的空间内数据为0x0113FCDC,又指了回来,所以大家不必在乎这块空间

    2. 大家注意观察应该会发现,我们从解决二义性到解决冗余共展示了两次菱形虚拟继承的内存情况,其中第一次是先有B类空间,其次是C类,而现在是先有C类空间,其次是B类(如上方内存图),这是因为两次查看内存所使用的代码中,我将D类的继承顺序调整

      //第一次
      class D :public B,public C;
      //第二次
      class D :public C,public B;
      

      自然是先继承的类相当于先声明,所以先申请出空间,先调用构造函数 ,这一点看似普通确是笔试中经常会考到的点。

  3. 上述测试代码中为了方便观察,所以A类的成员数量不多,占用空间不多,当A类的成员变多时,占用的空间变大,那虚拟继承的优势也就彻底体现出来了。

    比方说在上面菱形继承的问题 最后代码冗余那块,我们测试了一下使用菱形继承和菱形虚拟继承时子类所占用的空间,就明显发现了占用空间变少。

承上启下:

通过调试观察菱形虚拟继承的内存,大家应该会发现,它的内存分布主要以继承的父类来分布,不同的父类所占据一块连续的内存,这正是我们在之前基类和派生类对象赋值转换 中提到的切片 运行的原理,在父类对象接收子类对象时,只复制或指向子类对象中特定的区域。

3.3笔试题

class A {
public:
	A(string s) { cout << s << endl; }
	~A(){}
};

class B :virtual public A
{
public:
	B(string s1,string s2):A(s1) { cout << s2 << endl; }
	~B(){}
};

class C : virtual public A
{
public:
	C(string s1, string s2) :A(s1) { cout << s2 << endl; }
	~C() {}
};

class D :public B, public C
{
public:
	D(string s1, string s2, string s3, string s4)
		:B(s1, s2)
		, C(s1, s3)
		, A(s1)
	{
		cout << s4 << endl;
	}
	~D(){}
};

int main()
{
	D d("A", "B", "C", "D");

	return 0;
}

在这里插入图片描述

这道题一般是以选择题的形式给出,给出代码,选择最后的输出结果是什么,这里我们直接给出结果,在来分析一下为什么结果是这样的。

  1. 首先,我们观察代码可以发现这是一个菱形虚拟继承

  2. 既然是继承那一定是先调用最上层的类,也就是公共父类A,所以A是先打印的。最后调用最下层的类,也就是子类D,所以D是最后打印的。

    注意: 调用A的地方是在D的初始化列表中,而不再B和C类中,因为菱形虚拟继承解决了数据冗余和二义性,那A类的成员一定放在D类对象所属空间的最下方,所以直接在D类内调用A类的构造函数更好,使B和C类直接指向已经创建的A类即可。

  3. 既然是菱形虚拟继承,那所有的数据只有一份,也就是A类不会被调用多次,只会出现一次。

  4. 其次就是中间B和C的输出顺序了,在上面解决冗余的部分中我们提到,先继承的就是先声明的,而初始化列表的执行顺序于其出现拜访顺序无关,只与声明顺序有关,所以经常顺序就是构造函数的执行顺序。 所以是B先打印其次是C。

八.组合

下面这种在一个类中将另一个类的对象作为自己的成员变量的方式即为组合

class A {};

class B {
private:
	A _a;
};

继承与组合区别:

  1. public继承(最长见的基础,private和protected基础基本不会看到),是一种is-a 的关系。

    就是说每个派生类对象都是一个基类对象

  2. 组合是一种has-a 的关系。

    假设B组合了A,每个B对象中都有一个A对象。

  3. 继承允许你根据基类的实现来定义派生类的实现。这种通过派生类的复用通常被称为白箱复用(white-box reuse)。术语“白箱”是相对可视性而言:

    在继承方式中,基类的内部细节对子类可见。继承一定程度破坏了基类的封装,基类的改变(比如说修改访问限定符),对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高

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

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

    比如说:

    1. 一辆车一定是有轮胎的,那车和轮胎的关系就是组合,因为轮胎就是车的一部分(has-a )。
    2. 而在学生和人这两个类上,一定是用继承,因为学生就是人(is-a ),他需要继承来自人的所有的成员。

    考虑是否需要从新类上溯造型回基类。若必须上溯,就需要继承;否则用组合。

九.继承的总结和反思

  1. 很多人说C++语法复杂,其实多继承就是一个体现。 有了多继承,就存在菱形继承,有了菱形继承就有了菱形虚拟继承,底层实现就很复杂。所以一般不建议设计出多继承,一定不要设计出菱形继承。否则在复杂度及性能上都有问题。
  2. 多继承可以认为是C++的缺陷之一,很多后来的OO(object oriented面向对象)语言都没有多继承,如Java。
  3. 类之间的关系既可以用组合也可以用继承时,优先使用组合,

十.经典面试题

  1. 什么是菱形继承?菱形继承的问题是什么?
  2. 什么是菱形虚拟继承?如何解决冗余和二义性的
  3. 继承和组合的区别?什么时候用继承?什么时候用组合?

答案都在文中,这里不在一一讲解。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

React 中 TypeScript 和装饰器及 Hooks

概念 TypeScript 是强类型语言&#xff0c;相对于JavaScript 弱类型语言&#xff0c;它具有类型检测的功能&#xff0c;扩展了JavaScript 的语法。 TS的安装与执行&#xff1a; //全局安装typescript npm install typescript -g// 第二个因为 本来的node是不可能支持 ts那种民…

2023网络安全学习路线 非常详细 推荐学习

前言&#xff1a;首先咱们聊聊&#xff0c;学习网络安全方向通常会有哪些问题 目录&#xff1a; 1、打基础时间太长 学基础花费很长时间&#xff0c;光语言都有几门&#xff0c;有些人会倒在学习 linux 系统及命令的路上&#xff0c;更多的人会倒在学习语言上&#xff1b; …

SSD系列1——网络结构

SSD系列&#xff1a; SSD系列1——网络结构 SSD系列2——PriorBox SSD系列3——损失计算 SSD网络结构概述 SSD在VGGNet的基础上&#xff0c;增加了4个卷积模块&#xff0c;这些卷积模块获得的特征图具有不同的感受野&#xff0c;可以较好地检测不同尺度的目标。 VGG16 SSD网络…

springboot 断点上传、续传、秒传实现

文章目录 前言一、实现思路二、数据库表对象二、业务入参对象三、本地上传实现三、minio上传实现总结 前言 springboot 断点上传、续传、秒传实现。 保存方式提供本地上传&#xff08;单机&#xff09;和minio上传&#xff08;可集群&#xff09; 本文主要是后端实现方案&…

AI绘画:Lora模型训练完整流程!

关于AI绘画(基于Stable Diffusion Webui)&#xff0c;我之前已经写过三篇文章&#xff0c;分别是 软件安装&#xff0c;基本的使用方法&#xff0c;微调模型LoRA的使用。 整体来说还是比简单的&#xff0c;搞个别人的模型&#xff0c;搞个提示词就出图了。今天来一个有些难度…

推荐11个好用的prompt工具网站(附链接+论文)

同学们&#xff0c;你们prompt是自己苦哈哈码的吗&#xff1f;可别了&#xff0c;有现成的工具为啥不用&#xff1f; 今天我就和大家分享一些好用的prompt工具网站&#xff0c;用熟了ChatGPT、midjourney、stable diffusion能玩起来更爽&#xff01;搜罗了有十几个&#xff0c…

智能汽车实验二(视觉传感器标定)

实验二 视觉传感器标定&#xff08;实验报告&#xff09; 【实验目的】 1、了解开源图像处理库OpenCV的结构&#xff0c;掌握OpenCV的基本使用方法。 2、了解开源图像处理库OpenCV的基本模块功能&#xff0c;掌握常用图像处理方法。 3、掌握摄像机标定算法&#xff0c;学会使用…

Xilinx 7系列FPGA内置ADC

Xilinx 7系列FPGA全系内置了一个ADC&#xff0c;称之为XADC。这个XADC&#xff0c;内部是两个1mbps的ADC&#xff0c;可以采集模拟信号转为数字信号送给FPGA内部使用。 XADC内部可以直接获取芯片结温和FPGA的若干供电电压&#xff08;7系列不包括VCCO&#xff09;&#xff0c;用…

麒麟KylinV10SP1(2203)推荐安装一些硬件监控类软件与使用

目录 前言 1、tlp 电源管理 &#xff08;1&#xff09;查看电池容量、使用量、为Thinkpad设定电池充电开始结束阈值 &#xff08;2&#xff09;查看硬盘比如NVME SSD的型号种类、当前温度、读写量等信息&#xff1b; &#xff08;3&#xff09;查看CPU型号以及频率上下限、…

软件测试简单么,发展前景如何?

随着人工智能时代的到来&#xff0c;IT行业受到了越来越多人的重视。软件测试作为把控软件质量必不可少的环节&#xff0c;其重要性可见一斑。 软件测试可以说是算得上IT行业里相对简单的语言&#xff0c;但是也只是相对哈&#xff0c;如果想学习下去还是要看个人的学习能力的…

软件测试工作内容和职责有哪些

目前&#xff0c;在IT行业中测试的职位数量仅次于开发&#xff0c;可以说是第二大技术就业岗位。然而许多人对测试师工作的理解还停留在&#xff0c;只需要像用户一样使用产品&#xff0c;然后发现有问题提交报告就行了。其实这是极其不准确的&#xff0c;软件测试师在测试产品…

通过Dnspy调试解决powershell使用Install-module指定的转换无效的问题

之前运行Install-module -Name NtObjectManager出现以下错误&#xff1a; PackageManagement\Install-Package : Package NtObjectManager failed to be installed because: 指定的转换无效。 At C:\Program Files\WindowsPowerShell\Modules\PowerShellGet\1.0.0.1\PSModule.…

Shell编程之排序

目录 一、冒泡排序 二、选择排序 三、插入排序 基本思想&#xff1a; 四、反转排序 基本思想&#xff1a; 五、睡眠排序 六、希尔排序 基本思想&#xff1a; 举例 一、冒泡排序 冒泡排序&#xff0c;该排序的命名非常形象&#xff0c;即一个个将气泡冒出。冒泡排序一…

ChatDOC工具——使用ChatGPT高效阅读技术科研论文

ChatDOC是一款功能强大的人工智能阅读辅助工具&#xff0c;专为帮助用户快速理解论文内容而设计。使用ChatDOC&#xff0c;您可以通过上传PDF版论文文献&#xff0c;利用先进的ChatGPT技术&#xff0c;只需三个简单步骤&#xff0c;便可以高效地阅读论文&#xff0c;提高阅读效…

TypeScript初识

目录 介绍 定义 优点 类型声明 ts文件编译选项 自动编译 编译选项的各个属性 include compilerOptions 介绍 定义 TS&#xff08;TypeScript&#xff09;是一种由微软开发的编程语言&#xff0c;它是 JavaScript 的一个超集&#xff0c;提供了静态类型检查、类、接…

Linux系统编程——多线程[中]:互斥与同步

0.关注博主有更多知识 操作系统入门知识合集 目录 1.并发过程中的问题 2.互斥 2.1互斥锁 2.2如何看待互斥锁 2.3加锁和解锁的本质 2.4对锁做一个封装 2.5可重入函数与线程安全 2.6死锁 3.同步 3.1条件变量 1.并发过程中的问题 我们知道&#xff0c;同一个进程中的…

【SpringBoot】过滤器,监听器,拦截器介绍

文章目录 一、简介1、过滤器2、拦截器3、监听器 二、如何创建1、过滤器2、监听器3、拦截器 三、总结 一、简介 通过两幅图我们可以理解拦截器和过滤器的特点 1、过滤器 过滤器是在请求进入tomcat容器后&#xff0c;但请求进入servlet之前进行预处理的。请求结束返回也是&…

模拟IC与数字IC设计该怎么选?哪个岗位薪资高?

很多同学想要入行IC&#xff0c;但不知道数字和模拟方向怎么选&#xff1f; 如果没有亲身体会过模拟设计&#xff0c;并有发自内心的自信或者兴趣&#xff0c;一般不看好纯小白去学模拟电路设计。 模拟设计想做好&#xff0c;没有数学功底&#xff0c;没有电路分析的功底&…

面试题30天打卡-day24

1、Redis 为什么快&#xff1f; Redis 之所以快&#xff0c;主要是因为它具有以下特点&#xff1a; 纯内存操作&#xff1a;Redis 的数据存储在内存中&#xff0c;因此读写速度非常快&#xff0c;而无需像传统数据库一样从硬盘读取和写入数据。与此同时&#xff0c;Redis 支持…

【算法】动态规划算法求(编辑距离)

目录 编辑距离&#xff1a; 举例&#xff1a; 代码如下 调试&#xff1a; 核心代码&#xff1a; 画图演示上述代码&#xff1a; 编辑距离&#xff1a; 是一种计算两个自符串之间差异程度的方法&#xff0c;它通过比较两个字符串之间的插入&#xff0c;删除和 替换操作的数…