240607 继承
面向对象三大特性:封装、继承、多态
RE: 封装
- C++把数据和方法封装在类里面
- 迭代器和适配器
继承
1 基类 & 派生类
一个类可以派生自多个类,这意味着,它可以从多个基类继承数据和函数。定义一个派生类,我们使用一个类派生列表来指定基类。类派生列表以一个或多个基类命名,形式如下:
class 派生类(子类): 访问修饰符 基类(父类)注:访问修饰符是 public、protected 或 private 其中的一个,未使用它则默认是 private
假设有一个基类 Shape,Rectangle 是它的派生类,如下所示:
#include <iostream>
using namespace std;// 基类
class Shape
{public:void setWidth(int w){width = w;}void setHeight(int h){height = h;}protected:int width;int height;
};// 派生类
class Rectangle: public Shape
{public:int getArea(){ return (width * height); }
};int main(void)
{Rectangle Rect;Rect.setWidth(5);Rect.setHeight(7);// 输出对象的面积cout << "Total area: " << Rect.getArea() << endl;return 0;
}
2 访问控制和继承
访问 public protected private
同一个类 yes yes yes
派生类 yes yes no
外部的类 yes no no
3 赋值兼容转换(切片)
派生类对象 可以赋值给 基类的对象/基类的指针/基类的引用 。
(切片:把派生类中基类那部分切来赋值过去) 赋值语句:父类对象 = 子类对象
而 基类对象不能赋值给派生类对象 。
注:单独的语法规则,不是类型转换,没有产生临时变量,与下述引例机制不同
引例:截断和提升
// 类型转换会产生临时变量
int i = 1234;
printf("%x\n", i);// 4d2
// 截断
char ch = i;
printf("%x\n", ch);// ffffffd2
// 提升
i = ch;
printf("%x\n", i);// ffffffd2const int& ref_i = i;// 临时变量具有常性
printf("%d\n", ref_i);// -46const char& ref_ch = ch;// 临时变量具有常性
printf("%d\n", ref_ch);// -46
1234 的二进制表示为:10011010010(除 2 取余法)
补全至16位:0000 0100 1101 0010
然后,将每一组二进制转换为十六进制:
0000 = 0
0100 = 4
1101 = D(大小写皆可)
0010 = 2
最终组合成十六进制:4d2
第一个输出结果:4d2
截断:
因为 char 类型只占用 1 个字节(8 位),而 i 是一个 32 位的整数,在赋值给 char 时,只保留了最低 8 位的内容(1101 0010,即 0xd2),其余高位被截断。
留下的内容最高位是 1,表明这是一个负数,当 char 被提升为 int 以打印时,会进行符号扩展,高位会被填充为 1,使得 ch 变成 0xFFFFFFD2(在 32 位系统上)。
第二个输出结果:ffffffd2
提升:
ch 是有符号类型且其值为 1101 0010,
找到补码:当前的二进制数 11010010 就是补码。
求反码(将所有位取反):11010010 取反后得到:00101101
加 1:00101101 + 1 = 00101110
这个结果是 00101110,对应的十进制值是 46。
因为符号位是 1,表示这是一个负数,所以最终值为:-46。ch 是有符号类型且其值为 0xD2(-46 in decimal),类型提升时高位会填充符号位,因此 i 的值为 0xFFFFFFD2
第三个输出结果:ffffffd2
// 赋值兼容转换
Shape Sh1;
Rect1.name = "RECT";
Sh1 = Rect1;
Shape* ptr = &Sh1;
Shape& ref = Sh1;
ptr->name += "x";
ref.name += "x";
Rect1.PrintName();
Sh1.PrintName();
cout << endl;
两次打印结果如下:
RECT
RECTxx
对象的赋值是拷贝赋值;
Rect1 和 Sh1 是两个独立的对象: 由于 Sh1 是 Rect1 的副本,对 Sh1 的修改不会影响 Rect1,反之亦然。
4 继承的作用域
各作用域的影响:
作用域 | 语法编译查找规则 | 生命周期 |
---|---|---|
局部域 | ✅ | ✅ |
全局域 | ✅ | ✅ |
命名空间域 (默认不查找,除非展开或指定) | ✅ | 不存在 |
类域 | ✅ | 不存在 |
4.1 示例1
class Person
{
protected:string _name = "小李子"; // 姓名int _num = 111; // 身份证号
};class Student : public Person
{
public:void Print(){cout << "姓名:" << _name << endl;cout << "学号:" << _num << endl;// Student的_num和Person的_num构成隐藏关系cout << "身份证号:" << Person::_num << endl;}
protected:int _num = 999; // 学号
};void TestStu()
{Student s1;s1.Print();
};
姓名:小李子
学号:999
身份证号:111
4.2 示例2
class A {
public:void ft() {cout << "void ft()" << endl;}
};class B :public A {
public:void ft(int i) {A::ft();cout << "void ft(int i), i = " << i << endl;}
};void TestB() {B b;b.ft(1);// Q:重载,隐藏,编译报错,运行报错?// A:两者构成隐藏,函数重载的前提是在同一个作用域
}
void ft()
void ft(int i), i = 1
变式:
class A {
public:void ft() {cout << "void ft()" << endl;}
};class B :public A {
public:void ft(int i) {cout << "void ft(int i), i = " << i << endl;}
};void TestB() {B bb;bb.ft();// Q:重载,隐藏,重写,编译报错,运行报错?(不定项选择)// A:两者构成隐藏且编译报错// 如何调用父类?b2.A::ft();
}
5 继承过程中涉及的构造、拷贝构造和析构函数的工作机制
5.1 构造函数在继承中的作用和调用顺序
5.1.1 构造函数的调用顺序
当创建派生类对象时,基类的构造函数会先于派生类的构造函数被调用。这是因为派生类需要依赖基类的成员和功能,所以必须先初始化基类部分。
在构造派生类对象时,不能先初始化派生类再初始化基类,因为派生类的构造函数可能依赖于基类的成员。如果基类未先初始化,这些成员将包含随机值,从而导致不确定的行为。
示例:
#include <iostream>
using namespace std;class Base {
public:Base() {cout << "Base Default Constructor" << endl;}
};class Derived : public Base {
public:Derived() {cout << "Derived Default Constructor" << endl;}
};int main() {Derived d;return 0;
}
输出:
Base Default Constructor
Derived Default Constructor
解释:
- 基类成员当成一个整体:在构造派生类对象时,基类部分被当作一个整体,调用其默认构造函数来初始化。
- 派生类自己的成员:
- 内置类型成员:根据编译器的实现,可能会自动初始化(如置零),也可能不处理,产生未定义的值。
- 自定义类型成员:会调用它们的默认构造函数进行初始化。
5.1.2 派生类构造函数如何初始化基类
如果基类没有默认构造函数,或者需要传递参数,可以在派生类的构造函数的初始化列表中显式调用基类的构造函数。
示例:
class Base {
public:int baseValue;Base(int x) : baseValue(x) {cout << "Base Parameterized Constructor" << endl;}
};class Derived : public Base {
public:int derivedValue;Derived(int x, int y) : Base(x), derivedValue(y) {cout << "Derived Parameterized Constructor" << endl;}
};
解释:
- 派生类的构造函数在初始化列表中调用了基类的构造函数
Base(x)
,并初始化了自己的成员derivedValue(y)
。
5.2 拷贝构造函数在继承中的行为
5.2.1 默认拷贝构造函数的生成
当你没有显式定义拷贝构造函数时,编译器会为你生成一个默认拷贝构造函数。对于派生类,默认拷贝构造函数的行为如下:
- 基类成员当成一个整体:调用基类的拷贝构造函数来复制基类部分的数据。
- 派生类自己的成员:
- 内置类型成员:逐个成员进行值拷贝(浅拷贝)。
- 自定义类型成员:调用它们的拷贝构造函数进行复制。
示例:
class Base {
public:int baseValue;Base(int x) : baseValue(x) {}Base(const Base& other) : baseValue(other.baseValue) {cout << "Base Copy Constructor" << endl;}
};class Derived : public Base {
public:int* derivedValue;Derived(int x, int y) : Base(x) {derivedValue = new int(y);}// 默认拷贝构造函数// Derived(const Derived& other) : Base(other), derivedValue(other.derivedValue) {}~Derived() {delete derivedValue;}
};
当我们执行以下代码:
Derived d1(10, 20);
Derived d2 = d1; // 调用默认拷贝构造函数
可能的问题:
derivedValue
是一个指针,默认拷贝构造函数会进行浅拷贝,即复制指针的值。- 这会导致
d1
和d2
的derivedValue
指向同一块内存,可能在析构时造成重复释放(double free)等错误。
5.2.2 需要自定义拷贝构造函数的情况
当派生类的成员涉及到动态内存分配或需要深拷贝时,必须自定义拷贝构造函数。
示例(自定义拷贝构造函数):
class Derived : public Base {
public:int* derivedValue;Derived(int x, int y) : Base(x) {derivedValue = new int(y);}Derived(const Derived& other) : Base(other) { // 调用基类的拷贝构造函数derivedValue = new int(*other.derivedValue); // 深拷贝cout << "Derived Copy Constructor" << endl;}~Derived() {delete derivedValue;}
};
解释:
- 基类成员当成一个整体:在派生类的拷贝构造函数中,显式调用了基类的拷贝构造函数
Base(other)
。 - 派生类自己的成员:
- 内置类型成员:如果有内置类型成员,默认会进行值拷贝。
- 自定义类型成员:需要手动编写代码来实现深拷贝,防止多个对象共享同一块内存。
5.3 析构函数在继承中的作用和调用顺序
5.3.1 析构函数的调用顺序
当销毁派生类对象时,析构函数的调用顺序与构造函数相反:
- 首先调用派生类的析构函数,清理派生类特有的资源。
- 然后调用基类的析构函数,清理基类部分的资源。
示例:
class Base {
public:~Base() {cout << "Base Destructor" << endl;}
};class Derived : public Base {
public:~Derived() {cout << "Derived Destructor" << endl;}
};int main() {Derived d;return 0;
}
输出:
Derived Destructor
Base Destructor
解释:
- 先销毁派生类部分,释放派生类特有的资源。
- 然后销毁基类部分,确保对象的所有资源都被正确释放。
5.3.2 虚析构函数的重要性
在涉及多态的情况下,如果你通过基类指针删除派生类对象,基类的析构函数必须是虚函数(virtual
),否则可能导致派生类的析构函数不被调用,造成资源泄漏。
示例:
class Base {
public:virtual ~Base() {cout << "Base Destructor" << endl;}
};class Derived : public Base {
public:~Derived() {cout << "Derived Destructor" << endl;}
};int main() {Base* ptr = new Derived();delete ptr; // 正确调用派生类和基类的析构函数return 0;
}
输出:
Derived Destructor
Base Destructor
解释:
- 基类的析构函数被声明为虚函数后,
delete
基类指针时,会先调用派生类的析构函数,再调用基类的析构函数。 - 如果基类析构函数不是虚函数,只会调用基类的析构函数,派生类的资源可能得不到释放。
5.4 总结与注意事项
5.4.1 默认构造函数的行为
- 基类成员:在派生类的构造过程中,基类部分被当作一个整体,调用基类的默认构造函数。
- 派生类的内置类型成员:编译器可能会自动初始化(如置零),也可能不处理,这取决于编译器实现。
- 派生类的自定义类型成员:会调用它们的默认构造函数进行初始化。
5.4.2 默认拷贝构造函数的行为
- 基类成员:调用基类的拷贝构造函数,复制基类部分的数据。
- 派生类的内置类型成员:逐个成员进行值拷贝(浅拷贝)。
- 派生类的自定义类型成员:调用它们的拷贝构造函数。
5.4.3 何时需要自定义拷贝构造函数
- 当派生类的成员涉及到动态内存分配、文件句柄、网络连接等需要深拷贝的资源时,必须自定义拷贝构造函数和赋值运算符,以正确管理资源,防止浅拷贝带来的问题。
5.4.4 赋值操作符的注意事项
- 类似于拷贝构造函数,赋值操作符在默认情况下也会进行浅拷贝。如果涉及到需要深拷贝的成员,应该自定义赋值操作符。
5.4.5 避免资源泄漏和悬垂指针
- 正确地管理对象的生命周期,确保析构函数能被正确调用,防止资源泄漏。
- 注意浅拷贝带来的悬垂指针问题(指针指向已被释放的内存)。
相关文章:

240607 继承
面向对象三大特性:封装、继承、多态 RE: 封装 C把数据和方法封装在类里面迭代器和适配器 继承 1 基类 & 派生类 一个类可以派生自多个类,这意味着,它可以从多个基类继承数据和函数。定义一个派生类,我们使用一个类派生列表…...

轻松应对意外丢失:高效电脑数据恢复指南!
有时候由于误操作、硬件故障、病毒攻击等原因,电脑里的重要文件可能会突然消失不见。面对这样的情况,很多人会感到手足无措。其实,借助专业的电脑数据恢复软件,我们可以较为轻松地找回丢失的数据。今天,我们就来介绍几…...

vue项目中播放rtsp视频流
一、下载webrtc-streamer 下载地址:https://github.com/mpromonet/webrtc-streamer/releases 根据设备型号下载对应的版本到本地直接解压就行,我下载的是webrtc-streamer-v0.8.6-dirty-Windows-AMD64-Release.tar版本。 双击webrtc-streamer.exe可执行文…...

tomcat部署web配置环境变量
在Tomcat中设置环境变量通常涉及以下步骤: 找到Tomcat的启动脚本(如catalina.sh或catalina.bat)。 在启动脚本中设置环境变量。 对于catalina.sh(Linux/Unix系统),你可以在文件顶部添加环境变量…...

数据仓库技术及应用(练习1)
1.创表 (1)customers.csv CREATE EXTERNAL TABLE IF NOT EXISTS customers ( customer_id int, customer_fname varchar(45), customer_lname varchar(45), customer_email varchar(45), customer_password varchar(45), customer_street …...

老板的“神助攻”:公司电脑监控软件
在当今的商业世界中,企业管理者都希望员工能全身心投入工作,为企业创造更多价值。然而,员工上班摸鱼的现象却让许多老板头疼不已。公司电脑监控软件的出现,为解决这一问题提供了可能。接下来,我们将详细介绍几款优质的…...

前端vue部署网站
这里讲解一下前端vue框架部署网站,使用工具是 xshell 和 xftp (大家去官网安装免费版的就行了) 服务器 我使用的阿里云服务器,买的是 99 一年的,淘宝有新手9.9 一个月服务器。可以去用,学生的话是有免费三…...

Unity3D 动画回调函数详解
在Unity3D中,动画回调函数是实现精细动画效果的重要工具。通过动画回调函数,我们可以在动画的特定时刻执行自定义代码,从而实现更加灵活和复杂的动画效果。本文将详细解释Unity3D中的动画回调函数,并提供相应的代码实现。 对惹&a…...

el-table表格表尾合计行,指定合计某几列,自定义合计方法
🤵 作者:coderYYY 🧑 个人简介:前端程序媛,目前主攻web前端,后端辅助,其他技术知识也会偶尔分享🍀欢迎和我一起交流!🚀(评论和私信一般会回&#…...

一款工具替你解决Mac电脑菜单栏图标杂乱问题
你的菜单栏是不是各种图标挤在一起?图标过多显得杂乱?刘海屏遮挡菜单栏图标?教你如何让你的菜单栏变的简洁美观 iBar,一款Mac上优秀的菜单栏管理工具,可以自主选择菜单栏图标隐藏,单独窗口聚合展示&#x…...

MySQL 基础入门教程
参考视频地址:一小时MySQL教程 bilibili SQL 基础 数据库分为关系型数据库和非关系型数据库 常见的关系型数据库: MySQL、PostgreSQL、Oracle、SQL Server等。 非关系型数据库: MongoDB(文档型数据库)、Redis&am…...

俏生元将传统膳食智慧融入现代生活,自然成分绽放健康光彩
近年来,当代女性健康食品市场正经历快速发展和显著变化。随着女性健康意识的提升,市场对专门针对女性健康的产品需求快速上升。女性消费者对健康的关注不再局限于表面,而是越来越注重内在健康和生活质量的提升。此外,中式养生文化…...

腾讯云推流播放相关
直播的在线人数是否有上限? 腾讯云直播默认不限制观看直播的在线人数,只要网络等条件允许都可以观看直播。如果用户配置了带宽限制,当观看人数过多、超出了限制带宽时新的用户无法观看,此情况下在线人数是有限制的。 如何使用播…...

UE5运行时动态加载场景角色动画任意搭配-相机及运镜(二)
通过《MMD模型及动作一键完美导入UE5》系列文章,我们可以把外部场景、角色、动画资产导入UE5,接下来我们将实现运行时动态加载这些资产,并任意组合搭配。 1、运行时播放相机动画 1、创建1个BlueprintActor,通过这个蓝图动态创建1个LevelSequence,并Play 2、将这个Bluep…...

@JsonAlias和@JSONField序列化和反序列化
com.fasterxml.jackson.annotation.JsonAlias("expressCode") com.alibaba.fastjson.annotation.JSONField(name "expressCode") 这两个注解分别属于不同的JSON序列化框架:Jackson 和 Fastjson,它们的用途是处理JSON字段的名称映射…...

k8s1.27部署ingress 1.11.2
k8s1.27部署ingress 1.11.2 要求: 1、使用主机网络。 2、多节点部署,以来标签:isingressistrue ingress1.11.2支持版本 官方参考链接: https://github.com/kubernetes/ingress-nginx/ 官网yaml https://raw.githubuserconten…...

【运维】自动化运维详解
目录 引言一、什么是自动化运维?二、自动化运维的优势三、自动化运维的关键组成部分详解3.1 监控与告警3.2 部署与配置管理3.3 备份与恢复3.4 安全管理 总结 引言 在当今信息技术飞速发展的时代,企业对IT基础设施的依赖日益增强,传统的人工运…...

线控底盘技术介绍
随着汽车工业的不断发展,传统的机械控制系统逐渐向电子控制系统转变。线控底盘(Drive-by-Wire Chassis)作为这一转变的重要组成部分,正在改变汽车的操控方式和驾驶体验。本文将全面介绍线控底盘的概念、组成、工作原理、优缺点、应…...

DOM对象
DOM概述 官方定义: DOM是W3C制定的一个规范(标准),(Document Object Model,文档对象模型),是提供了访问和操作网页中各元素的方法,让程序可以动态的修改或改变网页元素的内容、样式、结构。 DOM是W3C制定的一个规范…...

[SQL] 数据库图形化安装和使用
一 安装 1.1 图形化安装 下载DataGrip安装包 点击此处一直下一步即可。点击免费使用。 进去界面后,选择新建一个项目 点击加号,创建一个Mysql连接。输入Mysql的连接信息。点击DownLoad下载Mysql的驱动 接下来点击创建的mysq项目中后面的三个点,选择…...

springboot 前后端处理日志
为了实现一个高效且合理的日志记录方案,我们需要在系统架构层面进行细致规划。在某些情况下,一个前端页面可能会调用多个辅助接口来完成整个业务流程,而并非所有这些接口的交互都需要被记录到日志中。为了避免不必要的日志开销,并…...

C++11 简单手撕多线程编程
如何使用线程库 std::thread 创建线程 thread1.join(); 阻塞主线程 thread1.detach(); 线程分离 #include<iostream> #include<thread>void helloworld(std::string msg) {for (int i 0; i < 10000; i){std::cout << i << std::endl;}//std::cou…...

刷c语言练习题7(牛客网)
1、函数fun的声明为int fun(int *p[4]),以下哪个变量可以作为fun的合法参数() A、int a[4][4]; B、int **a; C、int **a[4] D、int (*a)[4]; 答案:B 解析:如果是fun的合法参数,那么其类型应该与定义函数fun中的参数类型…...

Web Worker和WebSocket
Web Worker和WebSocket协议都是Web开发中用于处理多线程和实时通信的技术,但它们的应用场景和工作原理有所不同。 Web Worker Web Worker是HTML5引入的一项技术,它允许JavaScript代码在后台线程中运行,从而实现真正的多线程处理。Web Worke…...

【LeetCode】动态规划—712. 两个字符串的最小ASCII删除和(附完整Python/C++代码)
动态规划—712. 两个字符串的最小ASCII删除和 前言题目描述基本思路1. 问题定义2. 理解问题和递推关系3. 解决方法3.1 动态规划方法3.2 空间优化的动态规划 4. 进一步优化5. 小总结 代码实现PythonPython3代码实现Python 代码解释 CC代码实现C 代码解释 总结: 前言 在字符串处…...

wordpress Contact Form 7插件提交留言时发生错误可能的原因
WordPress Contact Form 7 插件提交留言时发生错误可能有以下几种原因,并提供相应的解决方案: 1. 表单字段验证失败 原因: 用户输入的数据未通过表单字段的验证规则。 解决方案: – 检查表单字段的验证规则是否设置正确。 –…...

uibot发送邮件:自动化邮件发送教程详解!
uibot发送邮件的操作指南?uibot发送邮件的两种方式? 在现代办公环境中,自动化流程的引入极大地提高了工作效率。uibot发送邮件功能成为了许多企业和个人实现邮件自动化发送的首选工具。AokSend将详细介绍如何使用uibot发送邮件。 uibot发送…...

【PostgreSQL】PG数据库表“膨胀”粗浅学习
文章目录 1 为什么需要关注表膨胀?2 如何确定是否发生了表膨胀?2.1 通过查询表的死亡元组占比情况来判断膨胀率2.1.1 指定数据库和表名2.1.2 查询数据库里面所有表的膨胀情况 3 膨胀的原理3.1 什么是膨胀?膨胀率?3.2 哪些数据库元…...

力扣(leetcode)每日一题 871 最低加油次数 | 贪心
871. 最低加油次数 题干 汽车从起点出发驶向目的地,该目的地位于出发位置东面 target 英里处。 沿途有加油站,用数组 stations 表示。其中 stations[i] [positioni, fueli] 表示第 i 个加油站位于出发位置东面 positioni 英里处,并且有 f…...

ppt压缩文件怎么压缩?压缩PPT文件的多种压缩方法
ppt压缩文件怎么压缩?当文件体积过大时,分享和传输就会变得困难。许多电子邮件服务对附件的大小有限制,而在网络环境不佳时,上传和下载大文件可能耗时较长。此外,在不同设备上播放时,较大的PPT文件还可能导…...