【C++笔记】继承

继承是面向对象的三大特性之一

定义类时,下级别的成员除了拥有上一级的共性,还有自己的特性

6.1 继承的基本语法

作用:减少代码重复量

语法:class 子类 : 继承方式 父类

例:class MyPage : public BasePage

子类也称为派生类、父类也称为基类

派生类中的成员,包含两大部分:一类是从基类继承过来的,一类是自己增加的成员。从基类继承过来的表现其共性,新增的表现其个性

6.2 继承方式

继承方式一共有三种:

  • 公共继承
    • 父类中的公共权限成员,子类中也是公共权限
    • 父类中的保护权限成员,子类中也是保护权限
    • 父类的私有权限成员,子类无法访问
  • 保护继承
    • 父类中的公共权限成员,子类中也是保护权限
    • 父类中的保护权限成员,子类中也是保护权限
    • 父类的私有权限成员,子类无法访问
  • 私有继承
    • 父类中的公共权限成员,子类中是私有权限
    • 父类中的保护权限成员,子类中是私有权限
    • 父类的私有权限成员,子类无法访问
      在这里插入图片描述
6.3 继承中的对象模型

父类中所有非静态成员属性都会被子类继承下去。父类中私有成员属性,是被编译器隐藏了,因此是访问不到,但是确实被继承下去了。

//例:
class Base{
public:
	int m_A;
protected:
	int m_B;
private:
	int m_C;
};

class Son : public Base{
public:
	int m_D;
};

void test(){
	cout << "size of Son = " << sizeof(Son) << endl;
    //输出结果:16
}
6.4 继承中构造和析构顺序

子类继承父类后,创建子类对象,也会调用父类的构造函数

继承中的构造和析构顺序如下:

  • 构造:先构造父类,在构造子类
  • 析构:顺序与构造相反
//例:
class Base{
public:
	Base(){
		cout << "Base的构造函数" << endl;
	}

	~Base(){
		cout << "Base的析构函数" << endl;
	}
};

class Son : public Base{
public:
	Son(){
		cout << "Son的构造函数" << endl;
	}

	~Son(){
		cout << "Son的析构函数" << endl;
	}
};

void test(){
	Son s;
}
------------
//输出
Base的构造函数
Son的构造函数
Son的析构函数
Base的析构函数
6.5 继承同名成员处理方式
  • 访问子类同名成员,直接访问即可
  • 访问父类同名成员,需要加作用域

例:s.Base::m_As.Base::func()

//测试案例
#include <iostream>
using namespace std;

class Base{
public:
	Base(){
		m_A = 100;
	}

	void func(){
		cout << "Base func 函数调用" << endl;
	}

	int m_A;

};

class Son : public Base{
public:
	Son(){
		m_A = 200;
	}

	void func(){
		cout << "Son func 函数调用" << endl;
	}

	int m_A;
};

void test01(){
	Son s;
	cout << "Son : m_A = " << s.m_A << endl;
	cout << "Base : m_A = " << s.Base::m_A << endl;
}

void test02(){
	Son s;
	s.func();
	s.Base::func();
}

int main(){

	test02();

	system("pause");
	return 0;
}

注意事项:

如果子类中出现和父类同名的成员函数,子类中的同名成员会隐藏掉父类中所有同名成员函数(包括重载)。如果先要访问父类同名成员,需要加作用域

6.6 继承同名静态成员处理方式

静态成员和非静态成员出现同名,处理方式一致

  • 访问子类同名成员,直接访问即可
  • 访问父类同名成员,需要加作用域
#include <iostream>
using namespace std;

class Base{
public:
	static int m_A;
	static void func(){
		cout << "Base - static void func" << endl;
	}
};
int Base::m_A = 100;

class Son : public Base{
public:
	static int m_A;
	static void func(){
		cout << "Son - static void func" << endl;
	}
};
int Son::m_A = 200;

void test01(){
	//通过对象访问
	Son s;
	cout << "Son - m_A = " << s.m_A << endl;
	cout << "Base - m_A =  " << s.Base::m_A << endl;
	//通过类名访问
	cout << "Son - m_A = " << Son::m_A << endl;
	//第一个::代表通过类名方式访问	第二个::代表访问父类作用域下的成员
	cout << "Base - m_A = " << Son::Base::m_A << endl;
}

void test02(){
	//通过对象访问
	Son s;
	s.func();
	s.Base::func();
	//通过类名访问
	Son::func();
	Son::Base::func();
}

int main(){

	test02();

	system("Pause");
	return 0;
}
6.7 多继承语法

C++允许一个类继承多个类

语法:class 子类 : 继承方式 父类1, 继承方式 父类2...

//例:
class Son : public Base1, public Base2{

如果多继承中父类出现了同名情况,子类使用时要加作用域

//例:
cout << "Base1 - m_A = " << s.Base1::m_A << endl;
cout << "Base2 - m_A = " << s.Base2::m_A << endl;
6.8 菱形继承

概念

  1. 两个派生类继承同一个类
  2. 又有某个类同时继承这两个类
  3. 这种继承称为菱形继承

菱形继承的问题

  1. Son1继承了Base的数据,Son2也继承了Base的数据,当GrandSon使用数据时,就会产生二义性

    解决方法:两个父类因为相同的数据,需要加以作用域区分

    g.Son1::m_A = 100
    
  2. GrandSon继承来自Base的数据有两份,但实际上有一份就可以

    利用虚继承,解决菱形继承数据重复的问题。发生虚继承之后,数据只有一个,且可以不加作用域直接访问。

    在继承前加上关键字virtual。此时的Base称为虚基类。

    class Son1 : virtual public Base
    

参考:黑马程序员匠心之作|C++教程从0到1入门编程,学习编程不再难
链接https://www.bilibili.com/video/BV1et411b73Z

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Huffiee

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值