C++之继承初识(不包含虚拟继承)

2023-05-16

C++是一种面向对象的语言,而面向对象,有着三大特征——封装,继承,多态。

关于封装,在我的其它博客中已经有过简单的介绍了。这里我将简单叙述一下面向对象的三大特征之二——继承。

目录

什么是继承

继承的定义格式

定义格式

继承方式

基类与派生类对象的赋值转换

继承中的作用域

同名隐藏

继承下的默认成员函数

继承与友元

继承与静态成员

单继承与多继承

菱形继承


 

什么是继承

继承机制是面向对象程序设计中,使代码复用的最重要手段,它允许设计者在保持原有类的特性的基础上进行扩展,增加功能。

通过这种方法产生的新的类,被称为派生类;被继承的类。则称为派生类的基类。

继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。

继承是类设计层次的复用。 

继承的定义格式

定义格式

class [派生类] : [继承方式] [基类]

继承方式

与Java中的继承不同,C++中的继承有着三种继承方式,与C++的三种访问限定符一一对应。

基类成员/继承方式public继承protected继承private继承
public成员publicprotectedprivate
protected成员protectedprotectedprivate
private成员派生类中不可见

1.基类中的private成员无论以什么方式继承,在派生类中都是不可见的。所谓的不可见并不是指它没有被继承到派生类之中,基类的私有成员也会被继承到派生类之中,但是在语法上限制派生类对象不管实在类里还是类外都无法访问它。

2.保护成员限定符是由于继承才出现的。由于基类的private成员在派生类中不可被访问,所以如果某一个成员我们既需要在派生类中访问它,又要防止它在类外被访问,我们可以将它的访问权限定义为protected。

3.与成员变量的访问权限声明一样,类继承时也可以忽略对继承方式的指明。同样的,如果派生类使用的是class,默认继承方式为private;如果派生类使用的是struct,默认继承方式为public。

PS.在实际应用中,很少使用 protected / private 继承,一般都是使用 public 继承。因为 protected / private 继承的成员只能在派生类的类中使用,这导致实际应用中的拓展维护性不强。

基类与派生类对象的赋值转换

要了解C++中基类与派生类对象的赋值转换,我们首先要了解在继承的条件下,基类与派生类的成员是如何存储的:

首先,我们假设一个类A,类A是类B的基类,即类B是类A的派生类。

类A中,有着 a1、a2、a3三个变量;类B中,除继承自A的三个成员变量外,还有b1、b2两个成员变量。

 这些变量在内存中的存储大致如上图所示。通过程序验证如下:


 基类与派生类对象的赋值转换,即赋值兼容规则,一定是在public的继承方式下才满足的。

通过公有继承,派生类就完全继承了基类的所有功能,包括其成员的访问属性,也就可以通过赋值兼容规则进行一系列操作。否则,由于基类成员在派生类中访问属性的改变,编译器会产生如下报错:

 派生类对象可以赋值给基类的对象 / 基类的指针 / 基类的引用。

派生类从基类处继承了基类的所有成员变量,并且继承自基类的成员变量存储在整个子类对象所占的存储空间的前端。将子类(派生类)对象赋值给父类(基类)的对象 或 指针 或 引用,简单来看就像是将子类前部继承自父类的那部分切下来拷贝给父类。

基类对象不能赋值给派生类的对象。 

这一点是非常好理解的,派生类对象往往会比基类对象多一些成员,如若将基类对象赋值给派生类对象,派生类中总会有成员无法被初始化。难道要调用一部分的构造函数?这显然是不行的。

基类的指针可以通过强制类型转换赋值给派生类的指针。但是必须是基类的指针指向派生类的对象的时候才是安全的。

基类指针指向派生类对象,是实现多态的途径之一。通过函数覆盖与虚函数,基类指针指向不同的对象,函数实现的功能也会不同。但这种操作往往会有很大的风险,所以在有可替代方法存在的情况下最好不要进行这类强制类型转换。

继承中的作用域

继承体系中基类与派生类都有独立的作用域。

正是由于基类与子类隶属于不同的作用域,所以子类与父类中有同名成员时,它们并不会构成函数重载,因为函数重载的前提是在同一作用域。

在继承体系中,也即在不同作用域的情况下,子类成员会屏蔽父类成员对同名成员的直接访问,这种情况被叫做隐藏,也称为重定义。

对于类成员的重定义,还有着如下细节:

成员变量隐藏:与成员变量类型无关,只与成员变量名是否相同有关

成员函数隐藏:与函数参数列表无关,只与成员函数名是否相同有关

PS.在子类成员函数中,可以使用如下方式显式访问基类成员:

基类 :: 基类成员

同名隐藏

实际上上述的一系列隐藏关系就是所谓的同名隐藏,不过这个概念还是单列出来再具体地讲一下比较好,内容不多且重复。

如果成员变量同名,子类对象直接访问同名成员变量时,优先访问自己的,与变量的类型是否相同无关。基类同名成员无法通过子类对象直接访问。

如果成员函数同名,子类对象直接访问同名成员函数时,优先访问自己的,与变量的类型是否相同无关。基类同名成员无法通过子类对象直接访问。 

class A {
protected:
    int a1;
    int a2;
public:
    A() 
    :a1(10)
    ,a2(20)
    {}

    void f() {
        cout << "A" << endl;
    }

    void printA() {
        cout << "a1 = " << a1 << endl;
        cout << "a2 = " << a2 << endl;
    }
};

class B : public A {
    int a1;
    int b1;
public:
    B()
    :a1(100)
    ,b1(200)
    {}
    void f() {
        cout << "B" << endl;
    }

    void printB() {
        cout << "a1 = " << a1 << endl;
        cout << "b1 = " << b1 << endl;
        cout << "A::a1 = " << A::a1 << endl;
    }
};

int main() {
    A a;
    B b;
    a.printA();
    b.printB();
    return 0;
}

 

 可以看到,当我们直接访问派生类中的成员变量时,派生类默认访问的是其自身所定义的同名成员变量,只有当我们在访问时加上作用域限定符,才会精确地访问到基类中的同名成员变量。

int main() {
    B b;
    b.f();
    b.A::f();
    return 0;
}

同名成员函数同理。

当然,这里还是有着一定的小缺陷的。我们并没有涉及到上述的不同变量类型以及不同参数列表情况下的同名成员也会隐藏,但这并不是很难验证的事情。

继承下的默认成员函数

我们讲过,C++中的类在创建时会,如果用户不显式定义,会生成六个默认成员函数。

分别是:

用于构造与清理的:

        构造函数

        析构函数

用于拷贝复制的:

        拷贝构造函数

        赋值运算符重载

对取地址运算符的重载:
        取地址运算符重载

        const取地址运算符重载

而派生类既然是类,自然也有上述六种默认成员函数,同时,由于其隶属于继承体系,它的六种拷贝构造函数自然与平常情况下的默认成员函数略有不同。

在派生类中,其默认成员函数的生成与基类的默认成员函数息息相关,大致可总结如下:

1.派生类的构造函数必须调用基类构造函数初始化派生类中继承自基类的一部分。如果基类中没有默认构造函数,那么派生类中必须在初始化列表处显式调用。

class A {
protected:
    int _a;
public:
    A(int a) :_a(a) {}
};

class B : public A{
public:
    int _b;
};

int main() {
    B b;
    return 0;
}

 由上,可见,在继承体系下,当编译器发现基类没有默认构造函数时,会自动将为派生类构建的默认构造函数删除,从而导致了尝试调用已删除函数的报错。

2.派生类的拷贝构造函数必须调用基类的拷贝构造函数完成基类的拷贝初始化。

3.派生类的赋值运算符重载必须调用基类的赋值运算符重载。 

4.派生类的析构函数在调用完成后会自动调用基类的析构函数清理基类成员。这样可以保证派生类先清理派生类成员再清理基类成员的顺序。 

5.派生类对象初始化,先调用基类构造在调用派生类构造;派生类对象清理与之相反,先调用派生类析构再调用基类析构。

using namespace std;

class A {
public:
    A() {
        cout << "create A" << endl;
    }

    ~A() {
        cout << "destory A" << endl;
    }
};

class B : public A{
public:
    B() {
        cout << "create B" << endl;
    }

    ~B() {
        cout << "destory B" << endl;
    }
};

int main() {
    B b;
    return 0;
}

可见,与一般的类对象的创建相同,先构造的后析构。

继承与友元

在继承体系下,友元关系不能继承,即基类的友元并不能访问子类的私有及保护成员。

继承与静态成员

类的静态成员只是声明于类中,其存储并不与类的对象在同一片区域。类的静态成员保存于静态存储区。与类的对象多少无关,静态成员只存在一份。

在继承体系下,基类定义了static成员,则整个继承体系只存在一个这样的成员,即所有子类只有这一个成员。

class A {
public:
    static int a;
};

class B : public A{
public:
    void print() {
        cout << "B::a = " <<  a << endl;
    }
};

class C : public A {
public:
    void print() {
        cout << "C::a = " << a << endl;
    }
};

int A::a = 0;

int main() {
    A a1;
    B b1;
    C c1;
    b1.print();
    c1.print();
    a1.a++;
    b1.print();
    c1.print();
    return 0;
}


单继承与多继承

单继承:一个子类只有一个直接父类时,该关系称为单继承

单继承就像是我们生物学的生物分类, 界、门、纲、目、科、属、种,每一级分类下的生物都有着一些相同的“特性”,然后根据剩余“特性”的不同,又细致地划分到更下一级,更细致的类之中。

动物界class Animalia
脊索动物门class Chordata : public Animalia
亚门脊椎动物亚门class Vertebrata : public Chordata
哺乳纲class Mammalia : public Vertebrata
亚纲真兽亚纲class Eutheria : public Mammalia
灵长目class Primates : public Eutheria
人科class Hominidae : public Primates
人属class Homo : public Hominidae
智人种class Homo_sapiens : public Homo

多继承:一个子类拥有两个或以上直接父类时称这个关系为多继承。 

相比而言,多继承可能并不像单继承那样条理清晰。如果要做一个比喻的话,它更像我们从社会角度去看一个人。 恰如一个人可以有许多不同的身份,不同的职业。

在多继承种,一个派生类拥有多个基类,不同的基类按照其继承时声明的顺序在派生类对象种依次存储。

 可以通过自己编写一些简单的代码,通过查看监视证明。


菱形继承

菱形继承是一种特殊的多继承。我们可以简单而形象地将其描述为——殊途同归。

 可见,派生类assitant同时继承自student以及engineer两个基类,而这两个基类又都继承自同一个基类people。这种向上追溯导致同一个类的成员被重复继承的现象被称为菱形继承。

结合之前所讲述的继承体系下派生类成员在内存中的存储,我们不难发现,在菱形继承的情况下会导致一些来自父类的成员重复出现在派生类之中。

class A {
public:
    int a1;
};

class B1 :public A {
public:
    int b1;
};

class B2 : public A {
public:
    int b2;
};

class C : public B1, public B2 {
public:
    int c1;
};

int main() {
    C c;
    c.a1 = 1;
    //c.B1::a1 = 1;
    //c.B2::a1 = 2;
    return 0;
}

在我们直接访问有菱形继承带来的冗余部分时,编译器会报错并告知我们a1并不明确。 

当然,我们时可以如被注释掉的地方一般,通过指明具体的作用域来访问到具体来自于哪个父类的成员,但是,这样只是解决了访问成员不明确的问题,却没有涉及到如和解决那部分冗余的数据。

对此,C++提供了虚拟继承用以解决菱形继承的二义性以及数据冗余问题。

class 派生类名:virtual [继承方式] 基类名1,virtual [继承方式] 基类名2,…{
    派生类成员声明与定义;
};

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++之继承初识(不包含虚拟继承) 的相关文章

  • UDP协议及编程

    UDP协议 UDP是无连接的 xff0c 即发送数据之前不需要连接 xff0c 因此减少了开销和发送数据之间的时延 UDP使用尽最大努力交付 xff0c 即不保证可靠交付 xff0c 因此主机不需要维持复杂的连接状态表 UDP是面向报文的
  • 24.STM32的IO口扩展PCF8574

    1 IO口扩展芯片 PCF8574是一款带IIC总线 xff0c 可使大多数MCU实现远程I O 口扩展 该器件包含一个8位准双向口和一个IIC总线接口 xff08 通信接口IIC xff0c 2根线可以扩展为8个口 xff09 PCF85
  • 网络编程(二)基础预备知识掌握

    网络编程预备知识 socket 是一种编程接口也是一种文件描述符 xff08 套接字 xff09 可用于 TCP UDP IPX通信 socket的类型 流式套接字 xff08 SOCK STREAM xff09 xff1a 提供一种面向连
  • 使用USTC-TK2016工具对USTC-TFC2016数据集进行处理——报错解决记录

    USTC TK2016数据处理工具 xff1a https github com yungshenglu USTC TK2016 USTC TFC2016数据集 xff1a https github com yungshenglu USTC
  • C++避免头文件重复包含问题

    避免头文件重复包含的方法 xff0c 通常有两种做法 xff1a 条件编译和 pragma once 条件编译就是通常的 ifndef XXX H define XXX H endif ifndef XXX H 表示 xff0c 如果没有包
  • c++ 调用yolov3-yolov4

    ifdef WIN32 define OPENCV define GPU endif include lt iostream gt include lt windows h gt include 34 yolo v2 class hpp 3
  • 一文搞懂UART、RS232、RS485、TTL等常用的接口与协议

    常用的接口与协议 PC机常用的按照接口数量细分为A型 xff08 15针 xff09 xff0c B型 xff08 25针 xff09 xff0c C型 xff08 37针 xff09 xff0c D型 xff08 50针 xff09 xf
  • 解决头文件重复包含与结构体未定义的问题

    一 养成良好的编程习惯 1 保证h文件的纯洁性 xff1a 尽量一个c文件对应一个h文件 xff0c 不要h文件包含许多h文件 这样可以优化编译速度且避免出现h文件中某个结构体之类未定义 xff0c 先在上个h文件中使用的错误 2 对于变量
  • Nokia 5110液晶屏显示模块的使用与开发

    Nokia 5110液晶屏显示模块 我们先来看看他的参数 nbsp 在深入研究连接和示例代码之前 让我们首先看一下其Pinout nbsp RST nbsp 针复位显示 它是低电平有效引脚 您可以通过将其拉低来重置显示 您也可以将此引脚连接
  • mongodb的文档的分页查询

    统计查询使用count xff08 xff09 方法 xff1a 统计comment集合的所有的记录数 xff1a db comment count 分页列表查询 xff1a 可以使用limit xff08 xff09 方法来读取指定数量的
  • 前后端分离项目的部署

    本次项目的项目架构图 xff1a Nginx主要部署的是 项目的静态资源 xff0c 即前端项目 通过Nginx的反向代理 xff0c 将请求发给Tomcat服务器 然后获取数据通过MySQL的主从复制 xff0c 主库负责更新数据 xff
  • echarts基本用法

    目录 tooltip 设置提示框信息 图表的提示框组件 legend 图例组件 toolbox 工具箱组件 可以另存为图片等功能 grid 网格配置 grid可以控制线型图 柱状图 图表大小 xAxs 设置x轴的相关配置 y轴同理 seri
  • java实现UDP通信传输信息

    实现UDP通信要依靠 DatagramPacket对象进行实现 UDP协议的相关介绍 xff1a UDP传输分为 服务端 和客户端 服务端发送消息 客户端接收消息 xff0c 服务端需要知晓客户端的 IP和所监听的端口号 话不多说直接上代码
  • MySQL篇之动态建表。

    在日常开发中 xff0c 可能会出现 动态配置的一些情况 xff0c 此时存储动态配置的一些数据时就需要动态建表了 xff0c 家人们可以选则两种方案 一种是采用mybatis的mapper xml文件里面使用 语句进行创建 二就是使用da
  • IDEA 2020.2 配置Tomcat服务器

    1 创建一个工程 2 右键项目名称 xff0c 选择 add framwork support 3 选中Web Application xff0c 默认勾选创建web xml 目录结构如下 4 点这两个地方中的任意一个 xff0c 添加配置
  • Java笔记之markdown语法

    狂神说Java系列视频笔记 本文章是作者学习B站系列视频 狂神说Java 的记录笔记与心得 xff0c 创作不易 xff0c 希望能够得到您的支持 1 Markdown的基本语法与使用 Markdown是当下一种较为流行的一种写作方法 通过
  • Java之数组专题

    文章目录 Java基础之数组专题数组的定义数组的声明与初始化数组元素的访问内存分析数组的使用For Each 循环数组作方法入参冒泡排序 多维数组稀疏数组 Java基础之数组专题 本文章是作者学习B站系列视频 狂神说Java 与经典书籍 J
  • Java封装详解

    Java类和对象 本文章是作者学习B站系列视频 狂神说Java 与经典书籍 Java核心技术 的记录笔记与心得 xff0c 创作不易 xff0c 希望能够得到您的支持 Java的构造器 Java的构造器 在用Java自定义类时 xff0c
  • C++ primer plus第七章习题中遇到的cin与cin.get问题

    cin gt gt 与cin get 是cpp程序常用到的输入函数 xff0c 近日在编写一道简单的习题时 xff0c 对二者产生了一些疑问 xff08 题目来源 C 43 43 primer plus 中文版习题第七章第六题 xff09

随机推荐

  • Leetcode部分经典链表题解析(涉及链表的反转、排序、合并、移除元素、成环、相交等操作)

    链表相关问题 第206题 反转链表 要求 xff1a 将给定链表进行反转操作 xff0c 第一个结点作为尾结点 xff0c 第二个结点指向第一个节点 xff0c 以此类推 xff0c 使得原链表的尾结点作为答案的头结点 思路一 xff1a
  • Linux报错:terminate called after throwing an instance of ‘std::regex_error‘ what(): regex_error

    文章目录 1 报错 xff1a 2 源码 3 原因 xff1a 4 解决办法 xff1a 5 运行成功 xff1a 1 报错 xff1a Linux中测试cpp httplib时出现报错std regex error xff0c 但源码中并
  • Redis学习笔记(狂神说)

    狂神视频地址 xff1a https www bilibili com video BV1S54y1R7SB Nosql概述 为什么要用Nosql 1 单机Mysql的年代 DAL xff1a 数据库访问层 在90年代 xff0c 一个基本
  • gazebo地图构建

    搭建地图环境是gazebo的基础功能 打开gazebo 可以在终端输入指令 打开的时候一定要有sudo xff0c 不然有可能在后面保存的时候出现画面卡住不动的情况 span class token operator span sudo g
  • Linux Ubuntu18.04安装微信

    最近做双系统 xff0c 在Ubuntu里下载微信时发现微信没有光网里没有开发Linux版本的微信 xff0c 找到了一些使用网页版登录微信的教程 xff0c 按着网上的教程做下来会的到一个如下的微信图标 打开扫描二维码后无法登录 可以在其
  • 虚拟机Ubuntu18.04 使用usb_cam调用笔记本摄像头

    虚拟机搭载Ubuntu18 04调用笔记本的摄像头 xff08 踩坑以及解决方法 xff09 一 建立工作空间 xff08 略 xff09 这里我建立的工作空间名称是catkin ones 二 下载usb cam包并进行编译 git clo
  • 关于UDP双向通信原理解释与范例

    注 本文不提供UDP通信的头文件 OK Let s do it 首先 我们需要了解什么叫做UDP xff0c 之前博主有些过TCP的通信范例 xff0c 我们可以了解到TCP的通信是一个稳定的 xff0c 可以进行双边通信的方式 同样附带上
  • windows10引导盘修复

    Windows修复引导项 前几天做双系统 xff0c 在使用Easybcd制作引导项时误删win10原本的引导项 xff0c 导致无法开机 xff0c 但是我可以通过磁盘直接启动linux 记录以下修复过程 在Linux里使用工具检查恢复
  • 局部路径规划:DWA算法

    一 概述 DWA算法是全称是Dynamic Window Approach 是在ROS中应用比较广泛的局部路径规划算法 主要作用是接受全局路径规划器生成的路径 xff0c 里程计信息 xff0c 地图信息等 xff0c 通过局部路径规划器将
  • ORB_SLAM2地图保存

    ORB SLAM2地图保存 在安装好orb slam2后按照教程中的方法做了地图构建的实验 xff0c 但是当地图达到想要的标准之后 xff0c 却发现没有办法保存地图 xff0c 查看ORB SLAM2源码发现在System h中有如下一
  • ros仿真小车

    ros仿真小车 补全前面博客中缺少的一些部分关于前面博客中的robotcar 本文也可单独食用 xff09 创建工作空间并初始化 span class token function mkdir span p catkin ws src sp
  • 【2023电赛备赛】msp430f5529学习笔记(一)

    写在前 本人目前是大二在读生 xff0c 第一次参加电赛 xff0c 准备不充分 xff0c 结果熬了四天 xff0c 最后成绩却不如人意 有51和32的基础 xff0c 然后想立一个flag系统的学习一下主打超低功耗的msp430f552
  • C语言经典题:有1、2、3、4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少?

    include lt stdio h gt 通过for循环将变量i j k的取值锁定在1 xff0c 2 xff0c 3 xff0c 4之间 int main int num 61 0 int i 61 0 j 61 0 k 61 0 fo
  • 单词逆序输出(c语言)

    int main int l j 61 0 int tmp 61 0 存储输入字符串的数组 char arr 100 61 34 i love beijing 34 用来存储输出字符串的数组 char arr2 100 输入字符串 gets
  • 进程虚拟地址空间

    关键词 xff1a 进程虚拟地址空间 xff0c 进程描述符 xff0c 页表 xff0c 分段式 xff0c 段页式 在进入正式的内容之前 xff0c 我们先了解一个重要的概念 进程描述符PCB 在Linux操作系统中 xff0c 描述进
  • 简单了解函数调用过程

    什么是栈帧 C C 43 43 程序的基本组成部分是函数 当程序在运行时 xff0c 每个函数每次调用都会在调用栈上维护一个独立的栈帧 xff0c 栈帧中维持着函数所需的各种信息 栈帧也叫过程活动记录 xff0c 是编译器用来实现过程 函数
  • 错题汇总1

    1 以下程序的运行结果是 xff08 xff09 int main void printf 34 s 5 3s n 34 34 computer 34 34 computer 34 return 0 A computer puter B c
  • 使用C/C++制作信息管理系统(demo)

    要求 xff1a 在windows环境下使用Vistual studio以C C 43 43 语言编译一个具有基础框架的客户信息管理系统 必须使用到封装 继承 map容器 SQL数据库技术 我 是 分 割 线 未经过UI处理的基础系统功能效
  • 错题汇总2

    1 下列程序的打印结果是 char p1 15 61 34 abcd 34 p2 61 34 ABCD 34 str 50 61 34 xyz 34 strcpy str 43 2 strcat p1 43 2 p2 43 1 printf
  • C++之继承初识(不包含虚拟继承)

    C 43 43 是一种面向对象的语言 xff0c 而面向对象 xff0c 有着三大特征 封装 xff0c 继承 xff0c 多态 关于封装 xff0c 在我的其它博客中已经有过简单的介绍了 这里我将简单叙述一下面向对象的三大特征之二 继承