C++ 面向对象(继承)

三、继承

3.1 继承的概念

基于一个已有的类 去重新定义一个新的类,这种方式我们叫做继承

关于继承的称呼

一个类B 继承来自 类 A 我们一般称呼

A类:父类 基类

B类: 子类 派生类

B继承自A A 派生了B

示例图的语法

class vehicle // 车类
{
	
}

class car:public vehicle
{
	
}

3.2 继承的意义

1、可以实现代码复用,减少重复代码的劳动量

2、继承是实现多态的必要条件

3、继承本身就是为了实现多态的,顺便实现了代码复用

3.3 继承的基本语法

3.3.1 使用方法

3.3.1.1 基本语法
class 子类名称:继承方式 父类名称
{
	子类新增的成员;
}

class B:public A
{
}
3.3.1.2 基类对象初始化

在子类的构造函数中,通常需要在初始化列表中显式调用基类的构造函数,以完成对基类成员的初始化。例如:

在继承时,需要在子类的构造函数的初始化表中

显性的调用父类的构造函数 来完成对父类成员的初始化格式

构造函数名(int val_1 , int val_2): class_A(val_1 , val_2);

3.3.1.3 派生类中调用基类

在派生类中可以通过以下方式调用基类的成员:

类外访问:
    对象名.基类名::变量;
    对象名.基类名::函数名(函数的实参表);


类内访问:
    基类名::变量;
    基类名::函数名(函数的实参表);

3.3.2 示例代码一(基本语法1)

#include <iostream>
#include <string>

using namespace std;

class person
{
private:

public:
    string name;
    int id;
    int height;

public:
    person(string name , int id , int height)
        :name(name) , id(id) , height(height)
    {
        cout << "父亲构造函数" << endl;
    }
};

class man : public person
{
private:
    string name;
public:
    man(string name , int  id ,int height)
        : person(name , id ,height)
    {

    }
    void show_data()
    {
        cout << person::name << " " << person::id << " " << this->height << endl ; 
    }
};



int main(int argc, char const *argv[])
{
    man m("张三" , 1 , 131);
    m.show_data();
    return 0;
}

示例代码1 运行结果如下:

3.3.3 示例代码二(基本语法2)

#include <iostream>
#include <string>

using namespace std;

class person
{
private:

public:
    string name;
    string sex;
    int age;

public : 
    person(string name ,int age ,string sex)
        :name(name), age(age) ,sex(sex)
    {
        cout << "父类有参构造" << endl;
    }
    person()
    {
        name = "未设置姓名";
        age = -1;
        sex = "沃尔玛塑料袋";
        cout << "父类无参构造" << endl;
    }
    ~person()
    {
        cout << "我是父类的析构" << endl ;
    }

};


class student:public person
{
private:

public:
    int id;

public:
    student(string name , string sex ,int age ,int id)
        :person(name , age , sex ) , id(id)
    {
        cout << "子类有参构造" << endl;
    }

    student()
    {
        name = "未设置姓名";
        age  = -1;
        sex  = "沃尔玛塑料袋";
        id   = -1 ;
        cout << "子类无参构造" << endl;
    }
    ~student()
    {
        cout <<"我是子类的析构" << endl ;
    }
public:
    void show_data()
    {
        cout << "姓名 " << name << endl;
        cout << "年龄 " << person::age << endl;
        cout << "性别 " << this->sex << endl;
        cout << "学号 " << this->id << endl;
    }
};



int main(int argc, char const *argv[])
{
    student s1 ;

    s1.name ="张三";
    s1.person::age = 10;
    s1.sex = "男";
    s1.id = 10;
    s1.show_data();
    
    return 0;
}

示例代码2 运行结果如下:

 

3.4 继承的方式

3.4.1 类中的访问控制权限

类内子类类外
public
protected×
private××

而在继承中 也有三种访问方式:publicprotectedprivate

  • 公共继承 public
  • 保护继承 protected
  • 私有继承 private

访问关系如下图:

image-20241010215659071

3.4.2 示例代码三(继承方式

#include <iostream>
#include <string>

using namespace std;

class person
{
private:
    string name;
private:
    int age;
protected:
    string sex;

public:
    person(string name ,int age ,string sex)
        :name(name) , age(age) , sex(sex)
    {
        cout << "父类有参构造" << endl;
    }
    person()
    {
        name = "未设置姓名";
        age = -1 ;
        sex = "沃尔玛塑料袋";
        cout << "父类无参构造" << endl;
    }
    ~person()
    {
            cout << "我是父类的析构" << endl;
    }
};

class student : private person
{
private:

public:
    int id;
public:
    student(string name ,string sex ,int age ,int id)
        :person(name ,age ,sex),id(id)
    {
        cout <<"子类有参构造" << endl;
    }
    student()
    {
        cout << "子类无参构造" << endl;
    }
    ~student()
    {
        cout<< "我是子类的析构" << endl;
    }
public:
    void show_data()
    {

    }

};

class son : private student
{
private:
  
public:
    son()
    {
        student::id = 10;
    }
    ~son()
    {

    }
};


int main(int argc, char const *argv[])
{
    son s;

    while (1)
    {
        /* code */
    }
    
    return 0;
}

示例代码3  运行结果如下:

3.5 继承的底层解释

实际开发中 一般继承都采用 pubilc 继承方式

如果不写继承方式,默认使用的都是 private 方式继承

1、子类中会继承父类中所有成员 ,包括私有成员,只不过在子类中不能直接访问父类私有成员需要父类提供公有的函数来访问父类的私有成员

2、当父子类中出现了同名的函数时,访问来也不会冲突,即使形参不同,也不构成重载关系 愿意是两个函数不在同一个空间内,如果想访问父类的成员 需要加 类名 :: 来修饰

类外访问:
    对象名.基类名::变量;
    对象名.基类名::函数名(函数的实参表);
类内访问:
    基类名::变量;
    对象名.基类名::函数名(函数的实参表);

 3.5.1 示例代码四(继承模型)

#include <iostream>
#include <string>

using namespace std;

class person
{
private:
    string name;    // 姓名
    string sex;     // 性别
    int age;        // 年龄

public:
    void set_name(string name)
    {
        this->name = name;
        cout << name << endl;
    }
};

class student:public person
{
private:    
public:
    //1、子类如何访问父类的私有成员
    void set_data(string name , string sex , int age )
    {
        person::set_name(name);
    }
    //2、当父类 和 子类 函数名冲突时 咋办
    void set_name(string name)
    {
        person::set_name(name);
    }

    ~student()
    {
        person();
    }
};


int main(int argc, char const *argv[])
{
    student s1;
    s1.set_name("张三");

    student s2;
    s2.set_data("王五","男",11);
    
    person p1;
    p1.set_name("李四");
    return 0;
}

示例代码4 运行结果如下:

 

##拓展 在继承中析构和构造的区别

 #构造

1、父类的构造函数不会被子类继承

2、需要在子类的构造函数的初始列表中,显性的调用父类函数的构造函数
    完成对父类中继承过来的成员初始化
    
3、如果没有在子类的构造函数初始化列表中调用父类中的构造函数 则使用无参构造
    如果父类没有无参构造 会报错
    
4、构造函数的调用顺序是
    先调用父类函数的构造函数
    再调用子类函数的构造函数

#析构

1、父类的析构函数不会被子类继承
    
2、不管是否显性调用父类的析构函数,父类的析构函数都会被调用
    完成对父类中继承过来的成员的善后工作
    
3、子类的析构函数中 无需调用父类的析构函数
    
4、析构函数的调用顺序是
    先调用子类函数的析构函数
    再调用父类函数的析构函数




前情回顾: ​​​​​​​

第一篇

C++ 面向对象(封装)-CSDN博客

第二篇

C++ 面向对象(构造 & 析构函数)-CSDN博客

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

mfc操作json示例

首先下载cJSON,加入项目; 构建工程,如果出现, fatal error C1010: unexpected end of file while looking for precompiled head 在cJSON.c文件的头部加入#include "stdafx.h"; 看情况,可能是加到.h或者是.cpp文件的头部,它如果有包含头文件, #include &…

基于微信小程序的模拟考试系统设计与实现(LW+源码+讲解)

专注于大学生项目实战开发,讲解,毕业答疑辅导&#xff0c;欢迎高校老师/同行前辈交流合作✌。 技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;…

Android10.0定制服务 APK安装或者更新过自动打开APK

需求描述 当用户APK安装|更新 成功之后自动打开APK。 实现思路 编写服务 注册接受安装广播 并判断安装的APK包名是否为客户APK包名,并自动打开。 具体实现 diff --git a/android/build/make/core/tasks/check_boot_jars/package_whitelist.txt b/android/build/make/cor…

linux 下tensorrt的yolov8的前向推理(python 版本)的实现

一、yolov8的python实现的环境搭建 #通过pip安装 pip install ultralytics #通过git克隆GitHub仓库 git clone <https://github.com/ultralytics/ultralytics.git> cd ultralytics #安装依赖 pip install -r requirements.txt #执行推理 yolo predict model./yolov8n.pt …

AI News(1/21/2025):OpenAI 安全疏忽:ChatGPT漏洞引发DDoS风险/OpenAI 代理工具即将发布

1、OpenAI 的安全疏忽&#xff1a;ChatGPT API 漏洞引发DDoS风险 德国安全研究员 Benjamin Flesch 发现了一个严重的安全漏洞&#xff1a;攻击者可以通过向 ChatGPT API 发送一个 HTTP 请求&#xff0c;利用 ChatGPT 的爬虫对目标网站发起 DDoS 攻击。该漏洞源于 OpenAI 在处理…

【数据挖掘实战】 房价预测

本次对kaggle中的入门级数据集&#xff0c;房价回归数据集进行数据挖掘&#xff0c;预测房屋价格。 本人主页&#xff1a;机器学习司猫白 机器学习专栏&#xff1a;机器学习实战 PyTorch入门专栏&#xff1a;PyTorch入门 深度学习实战&#xff1a;深度学习 ok&#xff0c;话不多…

ElasticSearch 学习课程入门(一)

引子 前文已经介绍了windows下如何安装ES&#xff0c;接下来的文章我会边学习边记录。OK&#xff0c;那就让我们开始吧。 一、ES基础操作 1、预备知识 &#xff08;1&#xff09;RESTful REST 指的是一组架构约束条件和原则。满足这些约束条件和原则的应用程序或设计就是 …

QML Binding和Qt.binding详解

一、Binding详解 1、介绍 QML Binding是一种声明式的编程概念&#xff0c;它允许在Qt Quick应用中自动更新属性之间的关系。通过使用Binding&#xff0c;我们可以在属性之间建立依赖关系&#xff0c;当其中一个属性的值发生变化时&#xff0c;绑定的属性会自动更新。QML Bind…

【优选算法】3----快乐数

这是第三篇算法题&#xff0c;也是十分的不好想&#xff0c;虽然定位在简单&#xff0c;但我感觉还是有些难度的~ -------------------------------------begin----------------------------------------- 题目解析&#xff1a; 同样的&#xff0c;这道算法题&#xff0c;看题…

计算机网络 (52)秘钥分配

一、重要性 在计算机网络中&#xff0c;密钥分配是密钥管理中的一个核心问题。由于密码算法通常是公开的&#xff0c;因此网络的安全性主要依赖于密钥的安全保护。密钥分配的目的是确保密钥在传输过程中不被窃取或篡改&#xff0c;同时确保只有合法的用户才能获得密钥。 二、方…

Docker Load后存储的镜像及更改镜像存储目录的方法

Docker Load后存储的镜像及更改镜像存储目录的方法 Docker Load后存储的镜像更改镜像存储目录的方法脚本说明注意事项Docker作为一种开源的应用容器引擎,已经广泛应用于软件开发、测试和生产环境中。通过Docker,开发者可以将应用打包成镜像,轻松地进行分发和运行。而在某些场…

【三国游戏——贪心、排序】

题目 代码 #include <bits/stdc.h> using namespace std; using ll long long; const int N 1e510; int a[N], b[N], c[N]; int w[4][N]; int main() {int n;cin >> n;for(int i 1; i < n; i)cin >> a[i];for(int i 1; i < n; i)cin >> b[i…

unity——Preject3——摄像机动画

1.当预设体有改动时候记住要应用一下 2.创建摄像头动画步骤 3.动画相关设置 取消动作循环 4. Hax Exit Time 讲解 在Unity的Animator Controller中&#xff0c;"Has Exit Time" 是一个用于控制状态转换的选项。它决定了当前状态是否需要在转换到下一个状态之前完成…

从零开始:Gitee 仓库创建与 Git 配置指南

你好呀&#xff0c;欢迎来到 Dong雨 的技术小栈 &#x1f331; 在这里&#xff0c;我们一同探索代码的奥秘&#xff0c;感受技术的魅力 ✨。 &#x1f449; 我的小世界&#xff1a;Dong雨 &#x1f4cc; 分享我的学习旅程 &#x1f6e0;️ 提供贴心的实用工具 &#x1f4a1; 记…

磁盘阵列服务器和普通服务器的区别

磁盘阵列服务器&#xff08;RAID Server&#xff09;和普通服务器在多个方面存在显著区别。以下是它们的主要区别&#xff1a; 1. 存储架构 磁盘阵列服务器 RAID 技术&#xff1a;使用 RAID&#xff08;冗余独立磁盘阵列&#xff09;技术&#xff0c;将多个硬盘组合成一个逻…

光谱相机如何还原色彩

多光谱通道采集 光谱相机设有多个不同波段的光谱通道&#xff0c;可精确记录每个波长的光强信息。如 8 到 16 个甚至更多的光谱通道&#xff0c;每个通道负责特定波长范围的光信息记录。这使得相机能分辨出不同光谱组合产生的相同颜色感知&#xff0c;而传统相机的传感器通常只…

Kubernetes 集群中安装和配置 Kubernetes Dashboard

前言 上篇成功部署Kubernetes集群后&#xff0c;为了方便管理和监控集群资源&#xff0c;安装Kubernetes Dashboard显得尤为重要。Kubernetes Dashboard 是一个通用的、基于 Web 的 UI&#xff0c;旨在让用户轻松地部署容器化应用到 Kubernetes 集群&#xff0c;并对这些应用进…

C ++ 也可以搭建Web?高性能的 C++ Web 开发框架 CPPCMS + MySQL 实现快速入门案例

什么是CPPCMS&#xff1f; CppCMS 是一个高性能的 C Web 开发框架&#xff0c;专为构建快速、动态的网页应用而设计&#xff0c;特别适合高并发和低延迟的场景。其设计理念类似于 Python 的 Django 或 Ruby on Rails&#xff0c;但针对 C 提供了更细粒度的控制和更高效的性能。…

【K8S系列】K8s 领域深度剖析:年度技术、工具与实战总结

引言 Kubernetes作为容器编排领域的行业标准&#xff0c;在过去一年里持续进化&#xff0c;深刻推动着云原生应用开发与部署模式的革新。本文我将深入总结在使用K8s特定技术领域的进展&#xff0c;分享在过去一年中相关技术工具及平台的使用体会&#xff0c;并展示基于K8s的技术…

mysql查看binlog日志

mysql 配置、查看binlog日志&#xff1a; 示例为MySQL8.0 1、 检查binlog开启状态 SHOW VARIABLES LIKE ‘log_bin’; 如果未开启&#xff0c;修改配置my.ini 开启日志 安装目录配置my.ini(mysql8在data目录) log-binmysql-bin&#xff08;开启日志并指定日志前缀&#xff…