C++学习笔记(十五)

继承

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

有些类与类之间存在特殊的关系,例如下图中:

我们发现,定义这些类时,下级别的成员除了拥有上一级的共性,还有自己的特性。

这个时候我们就可以考虑利用继承的技术,减少重复代码

1. 继承的基本语法

例如我们看到很多的网站中,都有公共的头部、公共的底部、甚至公共的左侧列表,只有中心内容不同,接下来分别使用普通写法和继承写法来实现网页中的内容

普通写法(98行):

#include <iostream>using namespace std;// Java网页
class Java
{
public:void header(){cout << "首页、公开课、登录、注册...(公共头部)" << endl;}void footer(){cout << "帮助中心、交流合作、站内地图...(公共底部)" << endl;}void left(){cout << "Java、Python、C++...(公共左侧列表)" << endl;}void content(){cout << "Java视频..." << endl;}
};// Python网页
class Python
{
public:void header(){cout << "首页、公开课、登录、注册...(公共头部)" << endl;}void footer(){cout << "帮助中心、交流合作、站内地图...(公共底部)" << endl;}void left(){cout << "Java、Python、C++...(公共左侧列表)" << endl;}void content(){cout << "Python视频..." << endl;}
};// C++网页
class CPP
{
public:void header(){cout << "首页、公开课、登录、注册...(公共头部)" << endl;}void footer(){cout << "帮助中心、交流合作、站内地图...(公共底部)" << endl;}void left(){cout << "Java、Python、C++...(公共左侧列表)" << endl;}void content(){cout << "C++视频..." << endl;}
};void test()
{Java ja;ja.header();ja.footer();ja.left();ja.content();cout << "---------------------------------" << endl;Python py;py.header();py.footer();py.left();py.content();cout << "---------------------------------" << endl;CPP cpp;cpp.header();cpp.footer();cpp.left();cpp.content();	
}int main(int argc, char* argv[])
{test();return 0;
}

继承写法(80行):

#include <iostream>using namespace std;// 公共页面
class BasePage
{
public:void header(){cout << "首页、公开课、登录、注册...(公共头部)" << endl;}void footer(){cout << "帮助中心、交流合作、站内地图...(公共底部)" << endl;}void left(){cout << "Java、Python、C++...(公共左侧列表)" << endl;}
};// Java网页
class Java :public BasePage
{
public:void content(){cout << "Java视频..." << endl;}
};// Python网页
class Python :public BasePage
{
public:void content(){cout << "Python视频..." << endl;}
};// C++网页
class CPP :public BasePage
{
public:void content(){cout << "C++视频..." << endl;}
};void test()
{Java ja;ja.header();ja.footer();ja.left();ja.content();cout << "---------------------------------" << endl;Python py;py.header();py.footer();py.left();py.content();cout << "---------------------------------" << endl;CPP cpp;cpp.header();cpp.footer();cpp.left();cpp.content();	
}int main(int argc, char* argv[])
{test();return 0;
}

总结:

继承的好处:可以减少重复代码

语法:class A :public B

A类  称为子类  也称为  派生类

B类  称为父类  也称为  基类

派生类中的成员,包含两大部分:

一类是从基类继承过来的,一类是自己增加的成员

从基类继承过来的表现其共性,二新增的成员体现了其个性

2. 继承方式

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

继承的方式一种有三种:

1. 公共继承

2. 保护继承

3. 私有继承

3. 继承中的对象模型

父类中的私有成员也是被子类继承下去了,只是由编译器给隐藏后访问不到

利用开发人员命令提示工具查看对象模型

1. 找到要查看的类的文件所在的路径

2. cd到此路径下

3. 查看类的名字和文件的名字

4. cl /d1 reportSingleClassLayout类名 文件名

4. 继承中构造和析构的顺序

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

问题:子类和父类的构造和析构顺序是谁先谁后

#include <iostream>using namespace std;class Dad
{
public:Dad(){cout << "父类的构造函数......" << endl;}~Dad(){cout << "父类的析构函数......" << endl;}
};class Son :public Dad
{
public:Son(){cout << "子类的构造函数......" << endl;}~Son(){cout << "子类的析构函数......" << endl;}
};void test()
{Son son;
}int main(int argc, char* argv[])
{test();return 0;
}

总结:继承中 先调用父类的构造函数,再调用子类的构造函数,析构顺序与构造相反

5. 继承同名成员的处理方式

问题:当子类与父类出现同名的成员,如何通过子类对象,访问到子类或者父类中同名的数据呢?

访问子类同名成员 直接访问即可

访问父类同名成员 需要加作用域

#include <iostream>using namespace std;class Dad
{
public:int m_a;Dad(){m_a = 10;}void func(){cout << "Dad中func()的调用..." << endl;}void func(int a){cout << "Dad中func(int a)的调用..." << endl;}
};class Son :public Dad
{
public:int m_a;Son(){m_a = 20;}void func(){cout << "Son中func()的调用..." << endl;}
};void test()
{Son son;cout << son.m_a << endl; // 直接调用 调用的是子类中的同名成员属性cout << son.Dad::m_a << endl; // 调用父类中的同名成员属性需要加作用域son.func(); // 直接调用 调用的是子类中的同名成员函数son.Dad::func(); // 调用父类中的同名成员函数需要加作用域// 如果子类中出现和父类同名的成员函数,子类的同名成员函数会隐藏掉父类中所有同名成员函数// 如果想访问到父类中被隐藏的同名成员函数,需要加作用域son.Dad::func(100);
}int main(int argc, char* argv[])
{test();return 0;
}

总结:

1. 子类对象可以直接访问到子类中同名成员

2. 子类对象加作用域可以访问到父类同名成员

3. 当子类和父类拥有同名的成员函数,子类会隐藏父类中同名成员函数,加作用域可以访问到父类中同名函数

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

问题:继承中同名的静态成员在子类对象上如何进行访问

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

访问子类同名成员 直接访问即可

访问父类同名成员 需要加作用域

#include <iostream>using namespace std;class Dad
{
public:static int m_a;static void func(){cout << "Dad:static void func()" << endl;}static void func(int a){cout << "Dad:static void func(int a)" << endl;}
};
int Dad::m_a = 100;class Son :public Dad
{
public:static int m_a;static void func(){cout << "Son:static void func()" << endl;}
};
int Son::m_a = 200;
void test()
{// 1. 通过对象访问cout << "通过对象访问:" << endl;Son son;cout << "Son中的m_a=" << son.m_a << endl;cout << "Dad中的m_a=" << son.Dad::m_a << endl;// 2. 通过类名访问cout << "通过类名访问:" << endl;cout << "Son中的m_a=" << Son::m_a << endl;// 第一个::代表通过类名方式访问 第二个::代表访问父类作用域下cout << "Dad中的m_a=" << Son::Dad::m_a << endl;// 1. 通过对象访问cout << "通过对象访问:" << endl;son.func();son.Dad::func();son.Dad::func(100);// 2. 通过类名访问Son::func();// 第一个::代表通过类名方式访问 第二个::代表访问父类作用域下Son::Dad::func();Son::Dad::func(100);
}int main(int argc, char* argv[])
{test();return 0;
}

7. 多继承语法

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

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

多继承可能会引发父类中有同名成员出现,需要加作用域区分

C++实际开发中不建议用多继承

#include <iostream>using namespace std;class Base1
{
public:int m_a;int m_b;Base1(){m_a = 100;m_b = 200;}
};class Base2
{
public:int m_a;int m_c;Base2(){m_a = 1000;m_c = 300;}
};class Son :public Base1, public Base2
{
public:int m_d;Son(){m_d = 400;}
};void test()
{Son s;cout << "Base1:m_a=" << s.Base1::m_a << endl;cout << "Base2:m_a=" << s.Base2::m_a << endl;cout << "m_b=" << s.m_b << endl;cout << "m_c=" << s.m_c << endl;cout << "m_d=" << s.m_d << endl;
}int main(int argc, char* argv[])
{test();return 0;
}

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

8. 菱形继承

菱形继承概念:

两个派生类继承同一个基类

又有某个类同时继承这两个类

这种继承被称为菱形继承,或者钻石继承

菱形继承的问题:

1. 羊继承了动物的数据,驼也继承了动物的数据,当草泥马使用数据时,就会产生二义性

2. 草泥马继承自动物的数据继承了两份,但其实这份数据只需要一份就可以

#include <iostream>using namespace std;// 动物类
class Animal
{
public:int m_age;
};// 羊类
class Sheep:public Animal{};// 驼类
class Camel:public Animal{};// 羊驼类
class Alpaca :public Sheep, public Camel{};void test()
{Alpaca alpaca;alpaca.Sheep::m_age = 18;alpaca.Camel::m_age = 28;// 当菱形继承时,两个父类拥有相同数据,需要加以作用域进行区分cout << "alpaca.Sheep::m_age = " << alpaca.Sheep::m_age << endl; // 18cout << "alpaca.Camel::m_age = " << alpaca.Camel::m_age << endl; // 28
}int main(int argc, char* argv[])
{test();return 0;
}

这份数据只要一份就可以,菱形继承导致数据有两份,产生资源浪费

利用虚继承可以解决菱形继承的问题

在继承之前加上关键字virtual变成虚继承

Animal类被称为虚基类

#include <iostream>using namespace std;// 动物类
class Animal
{
public:int m_age;
};// 羊类
class Sheep:virtual public Animal{};// 驼类
class Camel:virtual public Animal{};// 羊驼类
class Alpaca :public Sheep, public Camel{};void test()
{Alpaca alpaca;alpaca.Sheep::m_age = 18;alpaca.Camel::m_age = 28;// 当菱形继承时,两个父类拥有相同数据,需要加以作用域进行区分cout << "alpaca.Sheep::m_age = " << alpaca.Sheep::m_age << endl; // 28cout << "alpaca.Camel::m_age = " << alpaca.Camel::m_age << endl; // 28cout << "m_age = " << alpaca.m_age << endl; // 28
}int main(int argc, char* argv[])
{test();return 0;
}

继承了两份指针,两份指针都指向同一份内容

总结:

1. 菱形继承带来的主要问题是子类继承两份相同的数据,导致资源浪费以及毫无意义

2. 利用虚继承可以解决菱形继承问题

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【TB作品】51单片机 实物+仿真-电子拔河游戏_亚博 BST-M51

代码工程。 http://dt4.8tupian.net/2/28880a66b12880.pg3这段代码是用于一个数字拔河游戏的嵌入式系统&#xff0c;采用了基于8051架构的单片机&#xff0c;使用Keil C51编译器。 主要功能包括&#xff1a; 数码管显示&#xff1a;使用了四个数码管&#xff08;通过P2的控制…

css的元素显示模式(有单行文字垂直居中,侧边栏等案例)

目录 1. 什么是元素的显示模式 2. 元素显示模式的类型 块元素 行内元素 行内块元素 3. 元素显示模式的转换 4.文字垂直居中 5.具体实现案例 1. 什么是元素的显示模式 定义&#xff1a;元素显示模式就是元素&#xff08;标签&#xff09;以什么方式进行显示&#xff0c;…

windows10-tdengine的安装及使用

win10-tdengine的安装及使用 一、下载及安装配置1.1 下载安装1.2 配置1.3 关闭防火墙和开放端口1.3.1 关闭防火墙1.3.2 开放6030端口 二、启动及关闭服务2.1 启动tdengine服务2.2 关闭tdengine服务2.2 开机自启动配置 四、可视化工具五、TDengine 命令行&#xff08;CLI&#x…

PythonStudio:一款国人写的python及窗口开发编辑IDE,可以替代pyqt designer等设计器了

本款软件只有十几兆&#xff0c;功能算是强大的&#xff0c;国人写的&#xff0c;很不错的python界面IDE.顶部有下载链接。下面有网盘下载链接&#xff0c;或者从官网直接下载。 目前产品免费&#xff0c;以后估计会有收费版本。主页链接&#xff1a;PythonStudio-硅量实验室 作…

【图像分类】【深度学习】【Pytorch版本】 ResNeXt模型算法详解

【图像分类】【深度学习】【Pytorch版本】 ResNeXt模型算法详解 文章目录 【图像分类】【深度学习】【Pytorch版本】 ResNeXt模型算法详解前言ResNeXt讲解分组卷积(Group Converlution)分割-变换-合并策略(split-transform-merge)ResNeXt模型结构 ResNeXt Pytorch代码完整代码总…

汽车标定技术(十三)--标定概念再详解

目录 1.概述 2.基于Flash的标定 3.基于RAM的标定 4.AUTOSAR基于指针标定概念 5.小结 1.概述 最近有朋友问到是否用overlay标定完数据就直接写在Flash中&#xff0c;其实不然&#xff0c;是需要关闭overlay然后通过XCP Program指令集或者UDS刷进Flash。 从这里看出&#…

HiEV洞察|蔚来NIO Phone的前途,藏在车手互联的技术栈里

作者 |三少爷 编辑 |张祥威 「我觉得三年才可以看我们做手机是对的还是错的。」 几天前&#xff0c;蔚来创始人、董事长兼CEO李斌回答媒体提问时如是说道&#xff0c;手机项目本身的投入产出确实需要三年&#xff0c;等到华为&#xff0c;小米&#xff0c;甚至苹果所有这些手…

考研英语一图表作文必背模版句

英语一的作文还是很靠日常积累的&#xff0c;依据潘赟老师的九宫格理论&#xff1a; 2——图画描述5——意义论证8——建议措施 这3个模块式最为核心也是最容易拉开分差的&#xff0c;对于时间有限的同志不建议忙下功夫浪费时间&#xff0c;而对于另外6个模块&#xff0c;还是…

单片机期末复习

前言 发现很多人都写了单片机原理及接口技术课后习题的答案&#xff0c;但是也就只写了答案而已&#xff0c;可能是他们觉得太简单的缘故吧&#xff0c;我这里对此进行一下我近段时间复习的总结&#xff0c;本篇博客只展示选择题、填空题和判断题的答案&#xff0c;仅供参考&a…

如何在项目管理中跟踪资源可用性?

项目是有计划、有安排的&#xff0c;在不超支预算的情况下按期执行项目&#xff0c;项目经理必须了解资源的可用性。这是资源管理流程的一部分。 什么是资源可用性 资源可用性是指了解执行项目所需的资源、何时需要以及在何种情况下需要。能够协调完成项目所需的资源&#xf…

【深度学习目标检测】四、基于深度学习的抽烟识别(python,yolov8)

YOLOv8是一种物体检测算法&#xff0c;是YOLO系列算法的最新版本。 YOLO&#xff08;You Only Look Once&#xff09;是一种实时物体检测算法&#xff0c;其优势在于快速且准确的检测结果。YOLOv8在之前的版本基础上进行了一系列改进和优化&#xff0c;提高了检测速度和准确性。…

使用 Taro 开发鸿蒙原生应用 —— 当 Taro 遇到纯血鸿蒙 | 京东云技术团队

纯血鸿蒙即将到来 在今年 8 月的「2023年华为开发者大会&#xff08;HDC.Together&#xff09;」上&#xff0c;华为正式官宣「鸿蒙Next」&#xff0c;这个更新的版本将移除所有的 AOSP 代码&#xff0c;彻底与 Android 切割&#xff0c;使其成为一个完全自主研发的操作系统&a…

【Hive】——DML

1 Load&#xff08;加载数据&#xff09; 1.1 概述 1.2 语法 LOAD DATA [LOCAL] INPATH filepath [OVERWRITE] INTO TABLE tablename [PARTITION (partcol1val1, partcol2val2 ...)]LOAD DATA [LOCAL] INPATH filepath [OVERWRITE] INTO TABLE tablename [PARTITION (partcol…

runCatching异常捕获onSuccess/onFailure返回函数,Kotlin

runCatching异常捕获onSuccess/onFailure返回函数&#xff0c;Kotlin fun test(a: Int, b: Int) {runCatching {a / b}.onSuccess {println("onSuccess: $it")return ok(it)}.onFailure {println("onFailure: $it")return fail(it)} }fun ok(o: Any) {prin…

Rust语言基础语法使用

1.安装开发工具: RustRover JetBrains: Essential tools for software developers and teams 下载: RustRover: Rust IDE by JetBrains 下载成功后安装并启动RustRover 安装中文语言包插件 重启RustRover生效

C# WPF上位机开发(日志调试)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 程序开发的过程中&#xff0c;调试肯定是少不了的。比如说&#xff0c;这个时候&#xff0c;我们可以设置断点、查看变量、检查函数调用堆栈等等。…

五、Java核心数组篇

1.数组 概念&#xff1a; ​ 指的是一种容器&#xff0c;可以同来存储同种数据类型的多个值。 ​ 但是数组容器在存储数据的时候&#xff0c;需要结合隐式转换考虑。 比如&#xff1a; ​ 定义了一个int类型的数组。那么boolean。double类型的数据是不能存到这个数组中的&…

【ProcessOn】流程图咏道图Axure自定义元件使用----含流程图案例

目录 一&#xff0c;ProcessOn 1.1 ProcessOn是什么 1.2 ProcessOn应用场景 1.3 流程图 1.4 泳道图简介 1.5 Process网址与界面简介 二&#xff0c;流程图案例 2.1 门诊流程图 2.2 住院流程图 2.3 药房药库流程图 2.4 会议OA流程图 三&#xff0c;Axure自定义元件 …

Unity | AVpro的最基础使用方法(视频播放插件)

一、 AVpro的使用方法 (一)准备播放器MediaPlayer 1. AVpro的播放器是MediaPlayer&#xff0c;在Heirarchy面板里创建 2.播放器里放视频 a.把视频放到StreamingAssets文件夹下 b.你就可以在MediaPlayer里面找到这个视频 c.选中以后&#xff0c;就会变成 这里点击播放可以播放…

Java 基础学习(九)API概述、Object、String、正则表达式

1 API概述 1.1 API概述 1.1.1 什么是API API(Application Programming Interface)&#xff0c;意为&#xff1a;应用程序接口。API就是已经写好的的程序或功能&#xff0c;程序要需要时可以直接调用&#xff0c;无需再次编写。 API可以大致分为如下几类&#xff1a; 编程语…