IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C++—继承 -> 正文阅读

[C++知识库]C++—继承

一、继承的概念及其定义

1.1 继承的概念

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

class Person//父类
{
public:
	void Print()
	{
		cout << "name:" << _name << endl;
		cout << "age:" << _age << endl;
	}
protected:
	string _name = "peter"; // 姓名
	int _age = 18; // 年龄
};
//Student 和 Teacher是两个子类
class Student : public Person
{
protected:
	int _stuid; // 学号
};
class Teacher : public Person
{
protected:
	int _jobid; // 工号
};
  1. 继承后父类的Person的成员(成员函数+成员变量)都会变成子类的一部分。这里体现出了Student和Teacher复用了Person的成员。
  2. 我们使用监视窗口查看Student和Teacher对象,可以看到变量的复用。
  3. 调用Print可以看到成员函数的复用。

在这里插入图片描述

1.2 继承定义

1.2.1 定义格式

下面person是父类,也叫基类,student是子类,也叫派生类。
在这里插入图片描述

1.2.2 继承方式和访问权限符

在这里插入图片描述

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

类成员/继承方式publicprotectedprivate
父类的public成员子类的public成员子类的protected成员子类的private成员
父类的protected成员子类的protected成员子类的protected成员子类的private成员
父类的private成员在子类中不可见在子类中不可见在子类中不可见

总结

  1. 基类private成员在派生类中无论以什么方式继承都是不可见的。这里的不可见是指虽然基类的私有成员被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面都不能去访问它
  2. 基类private成员在派生类中是不能被访问,如果基类成员不想在类外直接被访问,但需要在派生类中能访问,就定义为protected。可以看出保护成员限定符是因继承才出现的。
  3. 基类的其他成员在子类的访问方式为,取父子权限较小的那一个,权限大小关系:public > protected > private。 (比如父类成员为protected成员,子类继承方式为public,取两者较小的那一个,就是protected)
  4. 使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过最好显示的写出继承方式。

在这里插入图片描述

注意在实际运用中一般使用都是public继承,几乎很少使用protetced/private继承,也不提倡使用protetced/private继承,因为protetced/private继承下来的成员都只能在派生类的类里面使用,实际中扩展维护性不强。

二、基类和派生类对象赋值转化

?我们可以知道,父类赋值给父类,子类赋值给子类是没有问题的。但是如果父类赋值给子类,子类赋值给父类会出现什么问题呢?

class Person
{
protected:
	string _name; // 姓名 
	string _sex; // 性别 
	int _age; // 年龄 
};
class Student : public Person
{
public:
	int _No; // 学号 
};
int main()
{
	Person p;
	Student s;

	//1.子类对象可以赋值给父类对象/指针、引用
	p = s;
	Person* ptr = &s;
	Person& ref = s;

	//2.父类对象不能赋值给子类
	s = p;

	//3.父类的指针和引用可以强转赋给子类的指针
	Student* sptr = (Student*)&p;
	Student& rref = (Student&)p;
	return 0;
}

总结

  • 常说的切片,寓意就是把子类中,父类的那一部分切下来,赋值过去。(这里也叫赋值兼容,仅限于public继承),这里不存在类型的转换,是天然的语法支持行为。
  • 父类是不能够赋值给子类的。
  • 父类的指针可以通过强转类型赋值给子类,但是很危险,存在访问越界的风险。
    在这里插入图片描述

三、继承中的作用域

  1. 在继承体系中,父类和子类都有独立的作用域
  2. 子类和父类中有同名成员时,子类成员将屏蔽父类对同名成员的直接访问,这种情况叫做隐藏,也叫做重定义。(在子类成员函数中,可以显式指出要访问的函数,使用基类::基类成员)
class Person
{
protected:
	string _name = "小李子"; // 姓名 
	int _num = 111; // 身份证号 
};
class Student : public Person
{
public:
	void Print()
	{
		cout << " 姓名:" << _name << endl;
		cout << " 身份证号:" << Person::_num << endl;//指定域
		cout << " 学号:" << _num << endl;
	}
protected:
	int _num = 999; // 学号 
};
int main()
{
	Student s1;
	s1.Print();
	return 0;
}

父子类出现了相同的变量,子类对父类的_num变量会隐藏起来,想要访问父类的成员,必须指定类域。

  1. 如果是成员函数的隐藏,只要函数名字相同,就构成隐藏。
class A
{
public:
	void fun()
	{
		cout << "func()" << endl;
	}
};
class B : public A
{
public:
	void fun(int i)
	{
		cout << "func(int i)->" << i << endl;
	}
};

int a = 90;
int main()
{
	//int a = 900;
	//cout << ::a << endl;//局部优先,指定全局域  ::
	B b;
	b.fun(100);

	b.A::fun();//在子类对象中,指定域访问父类成员函数

	b.fun();//被隐藏了,调不动
	return 0;
}

注意:最好在继承里面不要定义同名的成员!

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

6个默认成员函数,默认的意思就是,我们不写,编译器自动生成的,那么在子类中编译器默认生成的,会干什么呢?

我们不写默认生成的子类的构造和析构怎么处理?
a:对于继承下来的成员,调用父类的默认构造和析构函数
b:对于自己的成员,根普通的类处理方式一样(内置类型不做处理,自定义类型调用它对应的默认构造和析构函数)

class Person
{
public:
	Person(const char* 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;
	}
protected:
	string _name; // 姓名 
};
class Student : public Person
{
public:
	//不写,调用父类的构造和析构
protected:
	int _num; //学号
	string _s="hello";
};
int main()
{
	Student s;
	return 0;
}

结果展示:
在这里插入图片描述
在这里插入图片描述

我们不写默认生成的子类的拷贝构造和operator= 怎么处理?
a:对于继承下来的成员,调用父类的拷贝构造和operator=
b:对于自己的成员,跟普通类一样处理

在这里插入图片描述

总结:原则上,继承下来的成员调用父类的处理,自己的成员按照普通类的处理即可。

什么情况下必须自己写?

  1. 父类没有默认构造,需要我们自己显示写构造。
  2. 如果子类有资源需要释放,就需要自己显示写析构。(比如有new的资源)
  3. 如果子类存在浅拷贝问题,就需要自己实现拷贝构造和赋值解决浅拷贝问题。

如果我们要自己写,如何自己写?

  1. 父类成员调用父类的对应构造、拷贝构造、operator=和析构处理
  2. 自己成员按普通类处理。
class Person
{
public:
	Person(const char* 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;
	}
protected:
	string _name; // 姓名 
};
class Student : public Person
{
public:
	Student(const char* name="zhangsan",int num=0)
		:Person(name)//不能这样写 _name(name)
		,_num(num)
	{}

	Student(const Student& s)
		:Person(s)//这里s会自动切片传给父类
		,_num(s._num)
	{}

	Student& operator=(const Student& s)
	{
		if (this != &s)
		{
			Person::operator=(s);//与父类的构成了隐藏,必须指定类域
			_num = s._num;
		}
		return *this;
	}
	~Student()
	{}
protected:
	int _num; //学号
};
int main()
{
	Student s;
	Student s1(s);

	s1 = s;

	return 0;
}

有一个细节知识点:
父子析构函数名字会被统一处理成为"destructor( )",那么父子的析构函数就会构成隐藏关系,当子类调用析构函数时,必须指定作用域。

~Student()
{
	Person::~Person();//指定域
}

在这里插入图片描述
?子类析构函数结束时,出了作用域会自动调用父类的析构函数,所以我们自己实现析构函数时,不需要显示调用父类的析构函数,这样才能保证先析构子类成员,后析构父类成员。(创建s对象时,里面继承的有父类的那一部分,会被先构造父类成员,后构造子类成员,析构时先析构子类成员,后析构父类成员)

五、继承与友元

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

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;//父类的友元不能访问子类的保护成员
}
int main()
{
	Person p;
	Student s;
	Display(p, s);
	return 0;
}

六、继承与静态成员

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

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; // 研究科目
};
int main()
{
	Person p;	//调用一次构造函数
	Student s;	//调用一次父类的构造函数
	Graduate g;	//调用一次父类的构造函数

	cout << Person::_count << endl;   // 3
	cout << Student::_count << endl;  // 3
	cout << Graduate::_count << endl; // 3

	cout << &Person::_count << endl;   // 地址都相同
	cout << &Student::_count << endl;  // 地址都相同
	cout << &Graduate::_count << endl; // 地址都相同
	return 0;
}

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

单继承

一个子类只有一个直接父类时称为单继承
在这里插入图片描述

多继承

一个子类有两个或者两个以上的直接父类时称为多继承
在这里插入图片描述

菱形继承

菱形继承是多继承的一种特殊情况
在这里插入图片描述

菱形继承的问题数据冗余和二义性
从下面的对象成员来看,在Assistant的对象中就有两份Person成员,访问_name时就不知道访问谁的,就会导致问题。
在这里插入图片描述

class Person
{
public:
	string _name; // 姓名 
};
class Student : public Person
{
protected:
	int _num; //学号 
};
class Teacher : public Person
{
protected:
	int _id; // 职工编号 
};
class Assistant : public Student, public Teacher
{
protected:
	string _majorCourse; // 主修课程 
};
void Test()
{
	// 这样会有二义性无法明确知道访问的是哪一个 
	Assistant a;
	a._name = "peter";
	// 需要显示指定访问哪个父类的成员可以解决二义性问题,但是数据冗余问题无法解决 
	a.Student::_name = "xxx";
	a.Teacher::_name = "yyy";
}

虚拟继承

虚拟继承就可以解决菱形继承的二义性和数据冗余问题。
如上面的关系,在Student和Teacher的继承Person时使用虚拟继承。

class Person
{
public:
	string _name; 
};
class Student : virtual public Person//使用virtual继承
{
protected:
	int _num;  
};
class Teacher : virtual public Person//使用virtual继承
{
protected:
	int _id;
};
class Assistant : public Student, public Teacher
{
protected:
	string _majorCourse; 
};
void Test()
{
	Assistant a;
	a._name = "张三";
}

在这里插入图片描述

注意:虚拟继承不要在其他地方去使用!

虚拟继承原理

简化模型方便观察,再借助内存窗口观察对象成员模型。

//简化模型
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;
};
int main()
{
	D d;
	d.B::_a = 1;
	d.C::_a = 2;
	d._b = 3;
	d._c = 4;
	d._d = 5;
	return 0;
}

不使用虚拟继承:
在这里插入图片描述
这里我们可以看到数据冗余!存在着两份A数据。

现在是使用虚拟继承:
在这里插入图片描述
?这里分析得出D对象将A放在了最下面,这个A是同时属于B和C的,那么B和C是如何找到公共的A呢?我们还发现,B和C中还存在两个指针,指向了一张表。这两个表叫做虚基表指针,这两个表叫做虚基表。虚基表中存放的是偏移量,这个偏移量是B和C公共A成员的偏移量或者相对距离。
?如图所示,14是用16进制表示的,换算成10进制就是20,我们通过虚基表指针查找到了偏移量20,用起始地址+偏移量就可以找到公共A成员变量存放的位置了。

八、继承的总结与反思

  1. 很多人说C++语法复杂,其实多继承就是一个体现。有了多继承,就存在菱形继承,有了菱形继承就有菱形虚拟继承,底层实现就很复杂。所以一般不建议设计出多继承,一定不要设计出菱形继承。否则在复杂度及性能上都有问题。
  2. 多继承可以认为是C++的缺陷之一,很多后来的语言都没有多继承,如Java。

继承和组合

  • public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。
  • 组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。

注意:优先使用对象组合,而不是类继承!

//继承与组合
//is-a
class A {
protected:
	int _a;
};
class B : public A {
protected:
	int _b;
};


//has-a
class C {
protected:
	int _c;
};
class D {
protected:
	C _obj;
	int _d;
};

谢谢观看!

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-05-16 11:12:18  更:2022-05-16 11:13:23 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 18:33:10-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码