C++学习笔记:继承

news2024/11/29 0:53:49

继承

  • 什么是继承?
  • 继承的写法
  • 基类和派生类的赋值转换
  • 继承中的作用域
  • 派生类的默认成员函数
  • 单继承,多继承,虚拟继承
  • is-a 和 has-a

什么是继承?

继承是C++语言面向对象的三大特性之一,是面向对象程序设计使代码可以复用的最重要的手段,基本都是在一个类的基础上为了增加更多的不同的特性,从而衍生出来的另一个类的作用.

比如:我创造了一个人的类,成员有:姓名,性别

class Person
{
public:
	void Print()
	{
		cout << _name<<endl<< _sex <<endl<< _id << endl;
	}

private:
	string _name = "cris";
	string _sex = "沃尔玛购物袋";
	string _id = "123456";

};

现在,我要写的项目需要多种身份的人,比如:学生,老师,家长,但是这些身份的人物都有 人 这个类的特征,如果我们再去把每个人物的这些特征写一遍就太麻烦了,那么我们可以直接在 人 这个类上增加我们想要增加的东西
例如 :学生 的类可以继承 人 这个类

class Student : public Person
{
protected:
	int _stuid; 
};

老师 这个类也可以继承 人 这个类

class Teacher : public Person
{
protected:
 int _jobid; 
};

那现在,我们以学生为例来创建一个对象:

	Student s;

进行调试:
在这里插入图片描述
我们可以看到,Student对象s不仅有自己的_stuid成员,还有父类Person的三个成员,这就是继承;

简单来说就是为了在创造多个对象,而这些对象有很多共同的类成员,那么就可以写一个类把这些共同的类成员封装成为一个初始的类,其他的对象再基于这个初始的类增加自己单独的特性,并且这个初始的类被称为父类/基类,而在他的基础上衍生出来的类被称为子类/派生类.

继承的写法

继承的写法十分简单.在实现子类的时候,在字类类名后面加上 ; 再加上继承方式和父类名即可
继承方式有三种: public protected private
例如:上面的Student类继承父类Person的时候就是以public方式继承

class Student : public Person

三种继承方式实际上就是能够继承的成员不同
比如,上面的例子中Student类继承父类Person的方式是public,那么Student类就可以调用从父类那里继承下来的public成员:

int main()
{
	Student s;
	s.Print();
}

Student对象明明自己没有写Print函数,但是却能够调用,这就是从父类那里继承下来的
在这里插入图片描述
注意点:

  1. 基类private成员在派生类中无论以什么方式继承都是不可见的,这里的不可见是指基类的私
    有成员还是被继承到了派生类对象中,但是语法上限制派生类对象不管在类里面还是类外面都不能去访问它

  2. 基类private成员在派生类中是不能被访问,如果基类成员不想在类外直接被访问,**但需要在派生类中能访问,就定义为protected。**可以看出保护成员限定符是因继承才出现的

  3. 在实际运用中一般使用都是public继承,几乎很少使用protetced/private继承

  4. 使用关键字class时默认的继承方式是private,使用struct时默认的继承方式是public,不过 最好显示的写出继承方式

基类和派生类的赋值转换

  1. 在C++的继承中,派生类对象 可以赋值给 基类的对象 / 基类的指针 / 基类的引用。这里有个形象的说法叫切片或者切割。就是把子类中父类的成员的那一部分赋值给一个父类对象
	Student s;
	//子类成员可以赋值给父类对象/指针/引用
	Person p = s;
	Person* p1 = &s;
	Person& p2 = s;
  1. 但是父类成员不能直接赋值给子类成员

继承中的作用域

  1. 在继承体系中基类和派生类都有独立的作用域。
  2. 子类和父类中有同名成员,子类成员将屏蔽父类对同名成员的直接访问,这种情况叫隐藏,也叫重定义。(在子类成员函数中,可以使用 基类::基类成员 显示访问)
    例如:
class Person
{
public:
	void Print()
	{
		cout << _name<<endl<< _sex <<endl<< _id << endl;
	}

protected:
	string _name = "cris";
	string _sex = "沃尔玛购物袋";
	string _id = "123456";
};

class Student : public Person
{
public:
	void Print()
	{
		cout << _name << endl << _sex << endl << _id << endl;
		//调用父类_id
		//cout << _name << endl << _sex << endl << Person::_id << endl;
	}

protected:
	
	string _id = "7890";
};

int main()
{
	
	Student s;
	s.Print();

	return 0;
}

此时的_id显示的是 s 的缺省值
在这里插入图片描述

  1. 需要注意的是如果是成员函数的隐藏只需要函数名相同就构成隐藏。
    上面的例子中的Print函数就构成隐藏

  2. 虽然可以,但是在实际中在继承体系里面最好不要定义同名的成员。

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

  4. 在C++11标准中,在类名后加关键字 final 之后,这个类就不能被继承

派生类的默认成员函数

一般情况下,实例化一个派生类时,构造函数和拷贝函数都是先调用父类的构造/拷贝构造再调用子类的构造/拷贝构造 , 但是父子类的析构函数构成隐藏关系,默认先调用子类析构,在调用父类析构
在这里插入图片描述
当然,子类在构造函数初始化的时候是需要显式调用父类去构造的;
例如:

class Person
{
public:
	Person(const char* name = "cris",
		   const char* sex = "沃尔玛购物袋" , 
		   const char* id ="123456" )
		:_name(name)
		, _sex(sex)
		, _id(id)
	{

	}
	void Print()
	{
		cout << _name<<endl<< _sex <<endl<< _id << endl;
	}

protected:
	string _name ;
	string _sex ;
	string _id ;
};

class Student : public Person
{
public:
	Student(const char* id = "7890")
		:Person()    // 这里需要调用父类名来进行父类的初始化	
		,_id(id)
	{}
	void Print()
	{
		cout << _name << endl << _sex << endl << _id << endl;
		//调用父类_id需要显示加上父类类名
		//cout << _name << endl << _sex << endl << Person::_id << endl;
	}

protected:
	
	string _id;
};

单继承,多继承,虚拟继承

单继承;一个子类只有一个直接父类时称这个继承关系为单继承
在这里插入图片描述
多继承:一个子类有两个或以上直接父类时称这个继承关系为多继承
在这里插入图片描述
多继承的一种特殊状态----菱形继承:
在这里插入图片描述
而菱形继承因为构造函数的原因 , 有数据冗余和二义性的问题。
例如:在Assistant的对象中,因为Student类和Teacher类都继承了Person类,因此Person成员会有两份。

而虚拟继承可以解决菱形继承的二义性和数据冗余的问题。
虚拟继承的用法:在需要被多种子类共同继承的父类前加上关键词 virtual
如上面的继承关系,在Student和Teacher的继承Person时使用虚拟继承,即可解决问题。
需要注意的是,虚拟继承不要在其他地方去使用。

class Person
{
public :
 string _name ; // 姓名
};
class Student : virtual public Person
{
protected :
 int _num ; //学号
};
class Teacher : virtual public Person
{
protected :
 int _id ; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected :
 string _majorCourse ; // 主修课程
};
void Test ()
{
 Assistant a ;
 a._name = "peter";
}

在使用虚拟继承后,Assistant类中的Person成员就只有一份了

但是一般情况下最好不要用菱形继承和虚拟继承

is-a 和 has-a

一般来讲,is-a指的是继承

class Person
{
public :
 string _name ; // 姓名
};
class Student : public Person
{
protected :
 int _num ; //学号
};

即每个派生类对象都是一个基类对象,但是权限是public,派生类是可以改变基类所有的成员

继承允许你根据基类的实现来定义派生类的实现。这种通过生成派生类的复用通常被称为白箱复用(white-box reuse)。术语“白箱”是相对可视性而言:在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高。

而has-a一般被称为组合,指的是一个类的成员包含了另外一个类:

class Person
{
public :
 void func()
 {}
 protected:
  string _name;
};

class Student 
{
protected :
 Person _name;
};

在这种情况下,Student类可以调用Person类的共有成员,但不能使用保护成员,这是权限的缩小,但是会安全很多;

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

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

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

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

相关文章

十九、FreeRTOS之FreeRTOS软件定时器

本节主要需要掌握以下内容&#xff1a; 1&#xff0c;软件定时器的简介&#xff08;了解&#xff09; 2&#xff0c;软件定时器的状态&#xff08;熟悉&#xff09; 3&#xff0c;单次定时器和周期定时器&#xff08;熟悉&#xff09; 4&#xff0c;软件定时器结构体成员介…

Unity中Batching优化的GPU实例化(2)

文章目录 前言一、GPU实例化的Shader准备步骤1、在Pass中声明实例化需要的变体2、UNITY_VERTEX_INPUT_INSTANCE_ID 在顶点着色器的输入(appdata)和输出(v2f可选)中添加(uint instanceID : SV_InstanceID). 前言 在上篇文章中&#xff0c;我们做了一些GPU实例化的前置准备&…

查看NVIDIA CUDA版本号的四种方法

查看CUDA版本号这里有三种方法。 nvidia-smi 运行nvidia-smi命令&#xff1a; 右上角可以看到CUDA版本号。 CUDA Toolkit 下载地址&#xff1a;https://developer.nvidia.com/cuda-downloads $nvcc -V nvcc: NVIDIA (R) Cuda compiler driver Copyright (c) 2005-2023 NV…

【数据结构 — 排序 — 交换排序】

数据结构 — 排序 — 交换排序 一.交换排序1.基本思想2.冒泡排序2.1.算法讲解2.2.代码实现2.2.1.函数定义2.2.2.算法接口实现2.2.3.测试代码实现2.2.4.测试展示 3.快速排序3.1.算法讲解3.2.各大算法分别单独实现3.2.1快速排序hoare版本3.2.2.快速排序hoare改进版三数取中选key法…

四:爬虫-Cookie与Session实战

四&#xff1a;Cookie与Session实战 ​ 在浏览网站的过程中&#xff0c;我们经常会遇到需要登录的情况&#xff0c;有些页面只有登录之后才可以访问。在登录之后可以连续访问很多次网站&#xff0c;但是有时候过一段时间就需要重新登录。还有一些网站&#xff0c;在打开浏览器…

基于SSM框架的仓库管理系统

基于SSM框架的仓库管理系统 文章目录 基于SSM框架的仓库管理系统 一.引言二.系统设计三.技术架构四.功能实现五.界面展示六.源码获取 一.引言 现代商业环境中&#xff0c;仓库管理对于企业的运营效率和客户满意度至关重要。传统的手工管理方式已经无法满足日益复杂的仓储需求。…

51系列--基于MPX4250的压力计仿真设计

本文介绍基于MPX4250的压力计仿真设计&#xff08;完整仿真源文件及代码见文末链接&#xff09; MPX4250是一种线性度极强的一种压力传感器&#xff0c;它的线性范围为20-250kpa,测量范围也比较广泛&#xff0c;适用于大多数场合的压力检测。 仿真图如下 仿真运行视频 51系列…

项目记录:SpringBoot+Vue部署在阿里云服务器

目录 一、服务器配置 二、后端代码打包 三、前端项目打包 四、nginx配置 一、服务器配置 部署项目需要一个服务器&#xff0c;我们可以选择阿里云的云服务器ECS&#xff0c;在实例界面可以对服务器进行管理&#xff1a; 然后需要在mobaxterm配置jdk、mysql和nginx。注意配…

计算机毕业设计JAVA+SSM+springboot养老院管理系统

设计了养老院管理系统&#xff0c;该系统包括管理员&#xff0c;医护人员和老人三部分。同时还能为用户提供一个方便实用的养老院管理系统&#xff0c;管理员在使用本系统时&#xff0c;可以通过系统管理员界面管理用户的信息&#xff0c;也可以进行个人中心&#xff0c;医护等…

html网页设计 03表格

<!DOCTYPE html> <html><head><meta charset"utf-8"><title>表格-名字</title></head><body><!-- cellspacing:单元格与单元格之间的间距 &#xff0c;默认是2个像素间距cellspadding:单元格与内容之间的间距&am…

微信小程序制作-背单词的小程序制作

微信小程序–背单词的 好久没有发过文章了&#xff0c;但是不代表着我不去学习了喽&#xff0c;以下是我最近做的东西&#xff0c;前端的UI由朋友设计的&#xff0c;目前这个是前端使用的是微信小程序后端是Python的一个轻量型框架&#xff0c;FastApi&#xff0c;嗯&#xff…

解决方案:aarch64 ARM架构下安装Miniconda + 离线迁移Conda环境的全流程 踩坑避坑指南

目录 一、安装Miniconda1.1、确认本机架构1.2、下载Miniconda安装包1.3、安装Miniconda 二、离线配置Conda环境2.1、查看本机配置2.2、迁移虚拟环境2.3、可能遇见的报错 最后 在ARM架构下&#xff0c;由于Anaconda并不提供官方的ARM版本&#xff0c;在很多情况下强行在ARM系统中…

实战演练 | 在 Navicat 中格式化日期和时间

Navicat 支持团队收到来自用户常问的一个问题是&#xff0c;如何将网格和表单视图中的日期和时间进行格式化。其实这个很简单。今天&#xff0c;我们将介绍在 Navicat Premium 中进行全局修改日期和时间格式的步骤。 如果你想边学边用&#xff0c;欢迎点击 这里 下载免费全功能…

四十三、Redis基础

目录 一、认识NoSql 1、定义&#xff1a; 2、常见语法 3、与关系型数据库&#xff08;SQL&#xff09;的区别&#xff1a; 二、认识Redis 1、定义&#xff1a; 2、特征&#xff1a; 3、Key的结构&#xff1a; 三、安装Redis 四、Redis常见命令 1、数据结构介绍 2、…

孜然地址引导页V9(带后台)

刚刚在浏览之前经常访问的网站的时候我发现他不用那个域名了&#xff0c;然后我见这个页面好看&#xff0c;就把他干下来了&#xff0c;然后把给他写了个后台。另外如果你的子页面收录多的话&#xff0c;人家百度访问你的子页面会显示404的&#xff0c;所以为了流量可观安装这个…

改进的A*算法的路径规划(1)

引言 近年来&#xff0c;随着智能时代的到来&#xff0c;路径规划技术飞快发展&#xff0c;已经形成了一套较为 成熟的理论体系。其经典规划算法包括 Dijkstra 算法、A*算法、D*算法、Field D* 算法等&#xff0c;然而传统的路径规划算法在复杂的场景的表现并不如人意&#xff…

项目二 创建与操作学生管理数据库

项目二 创建与操作学生管理数据库 #目标 创建库&#xff1b;查看库&#xff1b;操作库&#xff1b;图形工具操作库1&#xff0c;创建学生管理数据库 #创建数据库 CREATE DATABASE [IF NOT EXISTS] db_name [[DEFAULT] CHARACTER SET charset_name] [[DEFAULT] COLLATE collat…

开发猿的平平淡淡周末---2023/12/10

天气阴 温度适宜17摄氏度 AM 昨晚竟然下小雨了&#xff0c;还好还好&#xff0c;昨天刷的两个背包基本干了 一觉睡到日三竿&#xff0c;谁是神仙&#xff0c;我是神仙&#xff08;哈哈哈哈哈哈&#xff09; 刷会儿视频 补充下起床的动力 洗漱&#xff0c;恰饭&#xff0c;肝…

springboot基础(80):redis geospatial的应用

文章目录 前言redis geospatial如何从地图上获取经纬度springboot 的相关方法调用准备redis服务器引用的依赖预设位置的keyGEOADD 添加位置GEORADIUS 获取指定经纬度附件的停车场&#xff08;deprecated&#xff09;GEORADIUS 获取指定成员附件的停车场&#xff08;deprecated&…

[LeetCode]-283. 移动零-1089. 复写零

目录 283. 移动零 描述 解析 代码 1089. 复写零 描述 解析 代码 283. 移动零 283. 移动零https://leetcode.cn/problems/move-zeroes/ 描述 给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 请注意 &…