05-C++ 类和对象-继承

类与对象-03

继承与派生

1. 继承的概念

c++最重要的特征是代码重用,通过继承机制可以利用已有的数据类型,来定义新的数据类型,新的类不仅拥有旧类的成员,还拥有新定义的成员。

一个 B 类继承于 A 类,或称从类 A 派生类 B。这样的话,类 A 成为基类(父类),类 B 成为派生类(子类)。

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

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

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

2. 继承的优点

1)减少重复的代码,减轻程序整体的体量。

2)继承的好处,可以将共性的内容封装成一个基类(父类),遇到专项业务时,可以扩展基类变为一个新类,在新类中重点扩展功能。

3. 继承的语法

class 子类:继承方式 父类名
{ 子类新增自己的数据和方法;
};

4. 继承方式

  • public(推荐):子类将 原封不动的继 承父类成员,但是 不能直接访问 父类 私有的成员

  • protected:子类 继承到 父类的成员 转换为 protected修饰的成员,但是 不能直接访问 父类 私有的成员

  • private:子类将 继承到所有父类成员 转换为 私有的成员,但是 不能直接访问 父类 私有的成员

在这里插入图片描述

示例1:public

#include <iostream>
using namespace std;
class A
{
public:int a;void testA(){cout << "testA" << endl;}
};
class B:public A
{
};
int main(int argc, char *argv[])
{B b;b.a = 10;b.testA();  //testAreturn 0;
}

示例2:protected ,若是protected继承,父类中的public将变为 protected,只能在当前类或子类中使用。

#include <iostream>
using namespace std;
class A
{
public:int a;void testA(){cout << "testA" << endl;}
};
class B:protected A
{
public:void testB(){cout << a << endl;testA();}
};
int main(int argc, char *argv[])
{B b;//此时是protected继承,父类中的public将变为 protected,只能在当前类或子类中使,//所以下面报错//b.a = 10;//报错//b.testA();//报错b.testB(); //93 testAreturn 0;
}

5. 注意

  1. 子类可以继承父类所有成员,但是父类私有成员不可访问
  2. 子类 不能继承父类构造函数拷贝构造析构函数,但是子类中可以调用
  3. 子类可以多继承
  4. 子类在 创建对象 时会 调用父类构造函数,如果 没有明确 写出 调用的父类构造 函数,默认调用 父类无参构造 ,此时如果父类 没有无参构造,程序报错
  5. 子类调用父类构造函数,在 子类构造函数后使用初始化列表方式调用父类构造函数
  6. operator=不能被继承

示例:

#include <iostream>
#include <cstring>
using namespace std;
class Anim{
private:char type[50];int age;char sex[10];
public:Anim(){cout << "Anim无参构造" << endl;}Anim(char *type,char *sex,int age){strcpy(this->type,type);strcpy(this->sex,sex);this->age = age;cout << "Anim有参构造" << endl;}Anim(const Anim& anim){strcpy(this->type,anim.type);strcpy(this->sex,anim.sex);this->age = anim.age;}void setType(char *type){strcpy(this->type,type);}char* getType(){return type;}void setSex(char *sex){strcpy(this->sex,sex);}char* getSex(){return sex;}void setAge(int age){this->age = age;}int getAge(){return age;}
};
class A{};
//3、子类可以多继承
class Yang:public Anim,public A
{
public:Yang(){}//5、初始化列表调用父类构造函数Yang(char *type,char *sex,int age):Anim(type,sex,age){}
};int main(int argc, char *argv[])
{//Yang y;//1、子类可以继承父类所有成员,但是父类私有成员不可访问//y.age;//2、子类不能继承父类的构造函数,拷贝构造,析构函数//Yang y("绵羊","公",18); //Anim有参构造//4、创建子类对象时会调用父类构造函数//默认调用父类无参构造//Yang y;//如果子类构造函数在初始化列表中指定调用父类的构造函数//那么就不会默认调用父类无参构造//Yang y("山羊","公",3);//如果父类中没有无参构造,子类没有明确写出调用父类构造函数//此时子类将调用父类无参构造,但是父类没有无参构造//程序报错Yang y;return 0;
}

6. 构造与析构的执行顺序

继承中的构造和析构:

  • 子类对象在创建时会首先调用父类的构造函数
  • 父类构造函数执行完毕后,才会调用子类的构造函数
  • 当父类构造函数有参数时,需要在子类初始化列表(参数列表)中显示调用父类构造函数
  • 析构函数调用顺序和构造函数相反

在这里插入图片描述

示例1:子类继承父类

#include <iostream>using namespace std;
class A{
public:A(){cout << "A的构造函数" << endl;}~A(){cout << "A的析构函数" << endl;}
};
class B:public A{
public:B(){cout << "B的构造函数" << endl;}~B(){cout << "B的析构函数" << endl;}
};
int main(int argc, char *argv[])
{B b;return 0;
}
//A的构造函数
//B的构造函数
//B的析构函数
//A的析构函数

示例2:子类继承父类,该子类中还要其他类的成员

#include <iostream>using namespace std;
class A{
public:A(){cout << "A的构造函数" << endl;}~A(){cout << "A的析构函数" << endl;}
};
class C{
public:C(){cout << "C的构造函数" << endl;}~C(){cout << "C的析构函数" << endl;}
};
class B:public A{
private:C c;
public:B(){cout << "B的构造函数" << endl;}~B(){cout << "B的析构函数" << endl;}
};
int main(int argc, char *argv[])
{B b;return 0;
}
//A的构造函数
//C的构造函数
//B的构造函数
//B的析构函数
//C的析构函数
//A的析构函数

7. 继承中父子类成员重名

7.1 成员变量重名

调用方式:

  • 操作子类成员变量:子类对象.成员变量名
  • 操作父类提供的成员变量:子类对象.父类名::成员变量名

示例:

#include <iostream>using namespace std;
class Basic{
public:int x;int y;Basic(int x):x(x){}
};
class Son:public Basic{
public:int x;//子类构造函数中必须调用父类构造函数//默认调用父类无参构造Son(int x1, int x2):Basic(x1){this->x = x2;}
};int main(int argc, char *argv[])
{Son son(10,100);//当父子类成员变量重名时//此时子类对象中有两个名称相同的变量//1个是继承父类的//1个是子类特有的//获取子类特有的该变量cout << son.x << endl;//100//获取父类提供的x变量cout << son.Basic::x << endl;//10//没有重名的成员直接获取cout << son.y << endl; //随机数return 0;
}
7.2 成员函数重名

概念:

  • 子类成员函数名父类成员函数名 重名,此时 子类该函数就是对父类名为该函数名的函数进行 重定义
  • 重定义
    • 继承关系中
    • 子类函数名与父类函数名相同
  • 特点:屏蔽父类该函数

调用方式:

  • 操作子类成员 函数:子类对象.成员函数名(实参列表)
  • 操作父类提供的成员函数:子类对象.父类名::成员函数名(实参列表)

示例:

#include <iostream>using namespace std;
class Fu{
public:void test(){cout << "fu test()" << endl;}void test(int a){cout << "fu test(int)" << endl;}void test(int a, int b){cout << "fu test(int, int)" << endl;}void fun01(){cout << "fu fun01()" << endl;}
};
class Zi:public Fu{
public:void test(){cout << "zi test()" << endl;}
};int main(int argc, char *argv[])
{Zi zi;//当父子类函数名重名时//子类调用重名函数,默认调用的是子类自己的函数zi.test();      //zi test()//子类调用继承于父类的 重名函数zi.Fu::test();  //fu test()zi.Fu::test(1);  //fu test(int)zi.Fu::test(10,20);  //fu test(int, int)//子类调用继承于父类的 非重名函数zi.fun01();     //fu fun01()return 0;
}

8. 多继承

概念:一个子类继承与 多个父类

语法:

class 子类名:继承方式1 父类1, 继承方式2 父类2, ...
{子类特有成员
}

父类构造顺序:

  • 子类对象创建时,按继承 编写的顺序 依次执行父类对象构造,与子类构造函数后初始化列表中的顺序无关

示例1:

#include <iostream>using namespace std;
class A{
public:int x;A(int a):x(a){}
};
class B{
public:int x;B(int a):x(a){}
};
class C:public A,public B
{
public:int x;C(int a,int b,int c):x(a),A(b),B(c){}
};
int main(int argc, char *argv[])
{C c(1,2,3);//获取c类中的xcout << c.x << endl;    //1//获取c类中继承与A类的xcout << c.A::x << endl; //2//获取c类中继承与B类的xcout << c.B::x << endl; //3return 0;
}

示例2:

#include <iostream>using namespace std;
class A{public:A(){cout << "A的构造函数" << endl;}~A(){cout << "A的析构函数" << endl;}};
class B{public:B(){cout << "B的构造函数" << endl;}~B(){cout << "B的析构函数" << endl;}
};
class C:public B,public A{public:C():A(),B(){cout << "C的构造函数" << endl;}~C(){cout << "C的析构函数" << endl;}
};
int main(int argc, char *argv[])
{C c;return 0;
}
//B的构造函数
//A的构造函数
//C的构造函数
//C的析构函数
//A的析构函数
//B的析构函数

9. 菱形继承(了解)

概念:

A的子类A1与A2B类多继承A1与A2类此时这种关系称为菱形继承

注意:

  • 菱形继承会导致子类用于多份祖先数据,当孙子类调用成员(函数或数据)时,会产生二义性。
  • 如:
    • A类中提供num成员变量
    • A1与A2类属于A的子类,,那么A1与A2类将各自拥有一份num
    • B作为A1与A2的子类,那么 B将拥有两个num

在这里插入图片描述

示例:

#include <iostream>
#include <cstring>
using namespace std;
class Anim{
public:char name[50];Anim(char *name){strcpy(this->name, name);}
};
class Yang:public Anim{
public:Yang(char *name):Anim(name){}
};
class Tuo:public Anim{
public:Tuo(char *name):Anim(name){}
};
class YangTuo:public Yang, public Tuo{
public:YangTuo(char *name01, char *name02):Yang(name01),Tuo(name02){}
};
int main(int argc, char *argv[])
{YangTuo yt("tom", "jerry");cout << yt.Yang::name << endl;  //tomcout << yt.Tuo::name << endl;   //jerrycout << &(yt.Yang::name) << endl;   //0x61fe2ccout << &(yt.Tuo::name) << endl;    //0x61fe5ereturn 0;
}

菱形继承类布局:

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

10. 虚继承

概念:使用 virtual修饰继承关系

语法:

class 子类名:virtual 继承关系 父类名
{};

解决问题:

解决菱形继承的调用二义性,多个类只保存一份相同数据

如:

  • A类中提供num成员变量
  • A1与A2类属于A的子类,那么A1与A2类将各自拥有一份num
  • B作为A1与A2的子类,那么B将拥有两个num,此时调用num会出现二义性
  • 所以只能虚继承使其存储一个num.

示例:

#include <iostream>
#include <cstring>
using namespace std;
class Anim{
public:char name[50];Anim(char *name){strcpy(this->name, name);}
};
class Yang:virtual public Anim{
public:Yang(char *name):Anim(name){}
};
class Tuo:virtual public Anim{
public:Tuo(char *name):Anim(name){}
};
class YangTuo:public Yang, public Tuo{
public:YangTuo(char *name01, char *name02):Anim(name01),Yang(name01),Tuo(name02){}
};int main(int argc, char *argv[])
{YangTuo yt("tom", "jerry");cout << yt.Yang::name << endl;  //tomcout << yt.Tuo::name << endl;   //tomcout << &(yt.Yang::name) << endl;   //0x61fe5ccout << &(yt.Tuo::name) << endl;    //0x61fe5creturn 0;
}

虚继承类布局:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

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

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

相关文章

搜索引擎推广的实践技巧提升你的品牌影响力-华媒舍

搜索引擎推广是一种有效提升品牌影响力的推广策略。通过关键词优化、广告创意设计、定向投放和数据分析与优化等实践技巧&#xff0c;可以提高品牌的知名度、点击率和转化率。在实施引擎霸屏推广之前&#xff0c;还需对实践效果进行评估&#xff0c;以确保推广策略的有效性和适…

Java IDEA JUnit 单元测试

JUnit是一个开源的 Java 单元测试框架&#xff0c;它使得组织和运行测试代码变得非常简单&#xff0c;利用JUnit可以轻松地编写和执行单元测试&#xff0c;并且可以清楚地看到哪些测试成功&#xff0c;哪些失败 JUnit 还提供了生成测试报告的功能&#xff0c;报告不仅包含测试…

以元旦为题的诗词(二)

都放假了吧&#xff0c;都有空了吧&#xff0c;可坐下来好好学学诗词&#xff0c;好好写些诗词了吧&#xff0c;我先来几首&#xff0c;你实在不行&#xff0c;去百度或者小程序搜索《美诗计》写一写 元旦 去年元日落寒灰&#xff0c;今岁清明在此杯 老眼看书如梦寐&#xff…

DM达梦数据库表占用空间大小

问题描述&#xff1a; 项目涉及用户量大且数据量大&#xff0c;为提高查询性能采用分表方式处理数据&#xff1b;根据业务要求总共4张业务表&#xff0c;每张业务表扩展成100张表&#xff0c;系统中总共400张表。部署至测试环境发现测试环境占用的磁盘空间是开发环境的8倍。 问…

nodeJS搭建免费代理IP池爬取贴吧图片实战

之前用python写过爬虫&#xff0c;这次想试试nodeJS爬虫爬取贴吧图片&#xff0c;话不多说代码如下&#xff0c;爬取制定吧的前十页所有帖子里的图片 爬取贴吧图片脚本 你得提前创建一个images文件夹 const axios require("axios"); const cheerio require("…

C语言编程入门 – 编写第一个Hello, world程序

C语言编程入门 – 编写第一个Hello, world程序 C Programming Entry - Write the first application called “Hello, world!” By JacksonML C语言编程很容易&#xff01; 本文开始&#xff0c;将带领你走过C语言编程之旅&#xff0c;通过实例使你对她颇感兴趣&#xff0c;一…

《异常检测——从经典算法到深度学习》25 基于深度隔离林的异常检测算法

《异常检测——从经典算法到深度学习》 0 概论1 基于隔离森林的异常检测算法 2 基于LOF的异常检测算法3 基于One-Class SVM的异常检测算法4 基于高斯概率密度异常检测算法5 Opprentice——异常检测经典算法最终篇6 基于重构概率的 VAE 异常检测7 基于条件VAE异常检测8 Donut: …

HarmonyOS4.0系统性深入开发08服务卡片架构

服务卡片概述 服务卡片&#xff08;以下简称“卡片”&#xff09;是一种界面展示形式&#xff0c;可以将应用的重要信息或操作前置到卡片&#xff0c;以达到服务直达、减少体验层级的目的。卡片常用于嵌入到其他应用&#xff08;当前卡片使用方只支持系统应用&#xff0c;如桌…

LTPI协议的理解——2、LTPI实现的底层架构

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 LTPI协议的理解——2、LTPI实现的底层架构 前言一、体系结构三、实现细节四、物理接口信号传输方法总结 前言 前面讲了LTPI的定义和大概结构&#xff0c;接下来继续理解LTPI…

openGauss学习笔记-178 openGauss 数据库运维-逻辑复制-逻辑解码-使用SQL函数接口进行逻辑解码

文章目录 openGauss学习笔记-178 openGauss 数据库运维-逻辑复制-逻辑解码-使用SQL函数接口进行逻辑解码178.1 前提条件178.2 操作步骤 openGauss学习笔记-178 openGauss 数据库运维-逻辑复制-逻辑解码-使用SQL函数接口进行逻辑解码 openGauss可以通过调用SQL函数&#xff0c;…

grafana 指标单位 bytes metric和bytes IEC

panel中数据size的单位&#xff0c;有两种&#xff1a;bytes metric、bytes IEC。 grafana中的bytes metric和bytes IEC的区别在于它们所使用的字节单位不同。bytes metric使用的是国际单位制&#xff08;SI&#xff09;中的字节单位&#xff0c;而bytes IEC使用的是IEC标准中…

idea远程开发环境搭建

idea远程开发环境搭建 一、安装包下载二、环境准备2.1服务器端jdk、maven安装&#xff0c;代码下载略2.2JetBrainsClients下载配置 三、远程环境配置3.1 创建项目3.2 填写服务器连接信息![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/4aa09073af5e4a66a5e83e7c5d1…