C++ 面向对象(继承)

embedded/2025/1/21 7:08:15/

三、继承

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博客


http://www.ppmy.cn/embedded/155707.html

相关文章

C#局部函数 VS Lambda表达式

一、引言 在 C# 的编程世界里&#xff0c;我们常常会遇到各种实现功能的方式&#xff0c;其中 Lambda 表达式和局部函数都是非常强大的特性。Lambda 表达式自诞生以来&#xff0c;凭借其简洁的语法和强大的功能&#xff0c;深受广大开发者的喜爱&#xff0c;尤其是在处理集合操…

【博客之星】2024年度总评选:技术总结—坚持分享,成就你我

总结 这一年&#xff0c;我总共写了81篇博文。 分布在前端、后端、运维、测试等各个方面。 这些无一例外&#xff0c;都是我个人的经验总结&#xff0c;或学习笔记。以前&#xff0c;我更多是写用自己的个人笔记中&#xff0c;以便需要的时候查阅回顾。后来&#xff0c;看到…

精选100+套HTML可视化大屏模板源码素材

大屏数据可视化以大屏为主要展示载体的数据可视化设计。 “大面积、炫酷动效、丰富色彩”&#xff0c;大屏易在观感上给人留下震撼印象&#xff0c;便于营造某些独特氛围、打造仪式感。 原本看不见的数据可视化后&#xff0c;便能调动人的情绪、引发人的共鸣。 使用方法&…

C# OpenCvSharp Yolov8 Face Landmarks 人脸特征检测

目录 介绍 效果 模型信息 项目 代码 下载 介绍 github地址&#xff1a;https://github.com/derronqi/yolov8-face yolov8 face detection with landmark 效果 模型信息 Model Properties description&#xff1a;Ultralytics YOLOv8-lite-t-pose model trained on w…

Springboot中使用Elasticsearch(部署+使用+讲解 最完整)

目录 引言 一、docker中安装Elasticsearch 1、创建es专有的网络 2、开放端口 3、在es-net网络上安装es和kibana 4、可能出现的问题 5、测试 6、安装IK分词器 7、测试IK分词器 二、结合业务实战 1、准备依赖 2、配置yml 3、读取yml配置 4、准备es配置类 5、编写测…

Wi-Fi 7、Wi-Fi 6 与 5G、4G 的全方位对比

随着无线通信技术的飞速发展&#xff0c;Wi-Fi 7、Wi-Fi 6&#xff0c;以及5G、4G 已经成为人们生活和工作中不可或缺的网络技术。无论是家庭网络、高速移动通信&#xff0c;还是工业物联网&#xff0c;这些技术都在发挥各自的作用。那么&#xff0c;它们之间有什么区别&#x…

使用numpy求解线性代数相关问题

在numpy中有numpy.array类型和numpy.mat类型&#xff0c;前者是数组类型&#xff0c;后者是矩阵类型。数组类型相乘是逐元素相乘&#xff0c;而矩阵类型相乘则是矩阵乘法。 以下使用numpy.array类型来进行线性代数问题求解。 矩阵的转置&#xff1a; A.T import numpy as n…

Spring Boot 3.4 正式发布,结构化日志!

1 从 Spring Boot 3.3 升级到 3.4 1.1 RestClient 和 RestTemplate 新增对 RestClient 和 RestTemplate 自动配置的支持&#xff0c;可用 Reactor Netty 的 HttpClient 或 JDK 的 HttpClient。支持的客户端优先级&#xff1a; Apache HTTP Components (HttpComponentsClient…