6.C++:继承

一、继承

//1.类中的保护和私有在当前类中没有差别;

//2.在继承后的子类中有差别,private在子类中不可见,所以用protected;

class person
{
public:void print(){cout << "name:" << _name << endl;cout << "age:" << _age << endl;}
protected:     //若换为private,则执行错误string _name = "peter";int _age = 18;
};class student: public person
{
protected:int stuid;
};
int main()
{student s;s.print();return 0;
}

二、 基类和派生类对象赋值转换

class person
{
protected:string _name;string _sex;int _age;
};
class student: public person
{
protected:int stuid;
};
int main()
{person p;student s;//子类和父类之间的赋值兼容规则//1.子类对象可以赋值给父类对象/指针/引用p = s;person* ptr = &s;person& ref = s;//反过来s = p;     //不可以student* ptr = &p; //有时候可以student& ref = p;return 0;
}

指针ptr指向student类中父类的那一部分;

引用ref是子类student中父类那一部分的别名; 


三、继承中的作用域

class person
{
protected:string _name = "peter";int _num = 111;
};
class student: public person
{
public:void print(){cout << "姓名:" << _name << endl;cout << "学号:" << _num << endl;}
protected:int _num = 999;
};
int main()
{student s;s.print();   return 0;
}
//输出: 姓名:peter// 学号:999

//1.两个_num在不同的作用域中,一个在父类,一个在子类,不会报错;

//2.当父类和子类同时有同名成员时,子类的成员隐藏于父类的成员(重定义);

//优先访问子类,现在子类中找,再去父类;

 访问父类_num:

class person
{
protected:string _name = "peter";int _num = 111;
};
class student: public person
{
public:void print(){cout << "姓名:" << _name << endl;cout << "学号:" << _num << endl;cout << "学号:" << person::_num << endl; //指定作用域即可}
protected:int _num = 999;
};
int main()
{student s;s.print();return 0;
}

 //1.A和B的fun构成什么关系?

//重定义(隐藏),不是重载,重载必须在同一个作用域

//函数只要函数名相同,不需要参数相同

class A
{
public:void fun(){cout << "func()" << endl;}
};
class B : public A
{
public:void fun(int i){A::fun();cout << "func(int i)->" << i << endl;}
};int main()
{B b;b.fun(); //执行错误b.fun(123456);b.A::fun(); //调用A的fun()return 0;
}

四、派生类的默认成员函数

class Person
{
public://1.构造函数Person(const char* name = "peter"): _name(name){cout << "Person()" << endl;}//2.拷贝构造Person(const Person& p): _name(p._name){cout << "Person(const Person& p)" << endl;}//3.赋值Person& operator=(const Person& p){cout << "Person operator=(const Person& p)" << endl;if (this != &p)_name = p._name;return *this;}//4.析构函数~Person(){cout << "~Person()" << endl;}
protected:string _name;
};class Student : public Person
{
public:Student(const char* name, int num): Person(name) //调用父类的构造函数, _num(num){cout << "Student()" << endl;}Student(const Student& s): Person(s) //调用父类的拷贝构造, _num(s._num){cout << "Student(const Student& s)" << endl;}Student& operator=(const Student& s){if (this != &s){Person::operator =(s);//重定义,调用父类的person::_num = s._num;}cout << "Student& operator= (const Student& s)" << endl;return *this;}~Student()  {// Person::~Person();//与~Person()构成隐藏cout << "~Student()" << endl;//调用子类析构后会自动调用父类析构}
protected:int _num; //学号
};
int main()
{Student s1("peter",11);Student s2(s1);return 0;
}

//输出:

Person()   //父类构造
Student()  //子类构造
Person(const Person& p)  //父类拷贝构造
Student(const Student& s)  //子类拷贝构造
~Student()   //s2析构
~Person()    //子类调用析构,自动调用父类析构
~Student()  //s1析构
~Person()

int main()
{Student s1("peter",11);Student s3("rose",20);s1 = s3;return 0;
}

 //输出:

Person()  //s1构造
Student()
Person()  //s2构造
Student()
Person operator=(const Person& p)  //父类赋值
Student& operator= (const Student& s)  //子类赋值
~Student()  //s3析构
~Person()
~Student()  //s1析构
~Person()


如何设计一个不能被继承的类? 

构造函数私有化,不能被继承,对象无法生成

class A
{
private:A(){}
};
class B :public A
{};

五、友元与继承

友元关系不能继承,也就是说基类友元不能访问子类私有和保护成员
class Student; //注意
class Person
{
public:friend void Display(const Person& p, const Student& s);
protected:string _name;
};
class Student : public Person
{friend void Display(const Person& p, const Student& s);
protected:int _stuNum; 
};void Display(const Person& p, const Student& s)
{cout << p._name << endl;cout << s._stuNum << endl;
}
int main()
{Person p;Student s;Display(p, s);return 0;
}

 或者:

class Student;
class Person
{
protected:string _name;
};
class Student : public Person
{friend void Display(const Person& p, const Student& s);
protected:int _stuNum; 
};void Display(const Person& p, const Student& s)
{cout << s._name << endl;cout << s._stuNum << endl;
}
int main()
{Person p;Student s;Display(p, s);return 0;
}

六、继承与静态成员

基类定义了static静态成员,则整个继承体系里面只有一个这样的成员。无论派生出多少个子类,都只有一个static成员实例 。
class Person
{
public:Person() { ++_count; }string _name; static int _count; 
};
int Person::_count = 0;
class Student : public Person
{
public:int _stuNum; 
};
int main()
{Person p;Student s;p._name = "jack";s._name = "rose";p._count = 1;s._count = 2;cout << Person::_count << endl;return 0;
}

_count放在静态区,只有一个_count,p和s中的_count是一个


七、复杂的菱形继承及菱形虚拟继承

单继承:一个子类只有一个直接父类时称这个继承关系为单继承
多继承:一个子类有两个或以上直接父类时称这个继承关系为多继承
菱形继承:菱形继承是多继承的一种特殊情况

菱形继承的问题:从下面的对象成员模型构造,可以看出菱形继承有数据冗余和二义性的问题。在Assistant的对象中Person成员会有两份。

class Person
{
public:string _name; // 姓名
};
class Student : public Person
{
protected:int _num; //学号
};
class Teacher : public Person
{
protected:int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:string _majorCourse; // 主修课程
};
void Test()
{// 这样会有二义性无法明确知道访问的是哪一个Assistant a;a._name = "peter";// 需要显示指定访问哪个父类的成员可以解决二义性问题,但是数据冗余问题无法解决a.Student::_name = "xxx";a.Teacher::_name = "yyy";
}

解决方法:

class Person
{
public:string _name; // 姓名
};
class Student : virtual public Person
{
protected:int _num; //学号
};
class Teacher : virtual public Person
{
protected:int _id; // 职工编号
};

virual分析:

 


八、继承和组合

//1.public继承是一种is-a的关系。也就是说每个派生类对象都是一个基类对象。
//2.组合是一种has-a的关系。假设B组合了A,每个B对象中都有一个A对象。
//3.继承与组合都可以的情况下,优先使用对象组合,而不是类继承 。
// Car和BMW Car和Benz构成is-a的关系
class Car {
protected:string _colour = "白色"; // 颜色string _num = "陕ABIT00"; // 车牌号
};class BMW : public Car {
public:void Drive() { cout << "好开-操控" << endl; }
};class Benz : public Car {
public:void Drive() { cout << "好坐-舒适" << endl; }
};
// Tire和Car构成has-a的关系
class Tire {
protected:string _brand = "Michelin"; // 品牌size_t _size = 17; // 尺寸
};
class Car {
protected:string _colour = "白色"; // 颜色string _num = "陕ABIT00"; // 车牌号Tire _t; // 轮胎
};

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

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

相关文章

#是啥,v-slot插槽的区别

在 Vue 3.x 中&#xff0c;您还可以使用 # 简写来代替 v-slot v-slot&#xff1a; v-slot 是 Vue 2.6 和 Vue 3.x 推荐的新的插槽语法。 v-slot 用于具名插槽&#xff0c;它允许您为插槽指定名称&#xff0c;并允许您传递具名插槽的内容。 例如&#xff0c;以下是使用 v-slot…

设计模式: 行为型之备忘录模式(13)

备忘录模式概述 备忘录模式&#xff08;Memento Pattern&#xff09;是一种行为设计模式&#xff0c;它允许在不破坏封装性的前提下捕获一个对象的内部状态&#xff0c;并在对象之外保存这个状态这样以后就可将该对象恢复到原先保存的状态。这种类型的设计模式属于行为模式在备…

【Qt 学习笔记】Qt常用控件 | 按钮类控件Push Button的使用及说明

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ Qt常用控件 | 按钮类控件Push Button的使用及说明 文章编号&#xff1…

2024年华中杯数学建模竞赛全攻略:ABC题思路解析+代码实现+数据集+论文撰写+全程答疑

引言 &#xff08;比赛后所有题目资料在文末获取呀&#xff09; 华中杯数学建模竞赛是数学建模领域的一项重要赛事&#xff0c;它不仅考验参赛者的数学建模能力&#xff0c;还考验了编程技能、数据分析能力和论文撰写能力。为了帮助参赛者更好地准备2024年的竞赛&#xff0c;本…

libftdi1学习笔记 6 - MPSSE QSPI

目录 1. 写 2. 读 3. 验证 QSPI采用4根线为数据口&#xff0c;SCK和CS保留同样的功能&#xff0c;一般&#xff14;个数据线采用MSB的方式发送数据&#xff0c;即高位在前。 QSPI只能是半双工工作。 1. 写 int qspiWriteBytes(uint8_t port, uint8_t* wrBuf, uint16_t l…

RAG学习笔记系列(一)

RAG 介绍 RAG 全称为 Retrieval Augmented Generation&#xff08;检索增强生成&#xff09;。是基于LLM构建系统的一种架构。 RAG 基本上可以理解为&#xff1a;搜索 LLM prompting。根据用户的查询语句&#xff0c;系统会先使用搜索算法获取到相关内容作为上下文&#xff0…

JavaScript如何设置定时器,怎么清除定时器

在JavaScript中&#xff0c;你可以使用setTimeout()和setInterval()函数来设置定时器&#xff0c;以及使用clearTimeout()和clearInterval()函数来清除定时器。 setTimeout() setTimeout()函数用于在指定的毫秒数之后执行一个函数或计算表达式。它返回一个表示定时器的ID&…

pnpm 报错: ERR_PNPM_META_FETCH_FAIL

今天突然遇到一个报错&#xff0c;pnpm 报错&#xff1a; ERR_PNPM_META_FETCH_FAIL  GET https://registry.npm.taobao.org/vue%2Fcli-service: request to https://registry.npm.taobao.org/vue%2Fcli-service failed, reason: certificate has expired问题原因&#xff1a;…

when语法学习及判断主机别名实例

目录 1、概念 2、实例配置 2.1 编写剧本体验when的用法 2.2 执行剧本文件 2.3 使用ansible临时命令进行测试 1、概念 when是用于判断的语法&#xff0c;将其用在每个动作的下方进行判断&#xff0c;使得只有在满足条件才会执行。 2、实例配置-判断主机别名 在每个客户端中…

移动端uniapp下载文件并保存到手机,下载准考证保存到手机

移动端uniapp下载文件并保存到手机&#xff0c;下载准考证保存到手机 1. 使用uni.request下载文件 uni.request({ url: http://example.com/file_url, // 要下载的文件的URL method: GET, responseType: arraybuffer, // 响应类型为arraybuffer (二进制形式) success: functi…

Compose 基础组件

文章目录 Compose 基础组件Modifier 修饰符Scaffold 脚手架 Compose 基础组件 Modifier 修饰符 在传统视图体系中&#xff0c;使用XML文件描述组件的样式&#xff0c;而在Compose中使用Modifier&#xff0c;每个基础的Composable组件都有一个modifier参数&#xff0c;通过Mod…

使用CCS软件查看PID曲线

在刚开始学习PID的时候&#xff0c;都需要借助PID的曲线来理解比例&#xff0c;积分&#xff0c;微分这三个参数的具体作用。但是这些曲线生成一般都需要借助上位机软件或者在网页上才能实现。如果是在单片机上调试程序的话&#xff0c;想要看曲线&#xff0c;一般就是通过串口…

【YOLOv5】使用yolov5训练模型时报错合集

文章目录 前言问题1 -- VsCode终端无法进入Anaconda创建的虚拟环境【问题描述】【问题分析】【解决方式】方法一方法二 问题2 -- 怎么在VsCode中为项目配置Anaconda创建的虚拟环境【问题描述】【解决方式】 问题3 -- yolov5训练模型时报错RuntimeError: result type Float cant…

web自动化测试系列-selenium常用方法定位(五)

目录 1.selenium的定位方法 2.操作案例 3.实现代码 前面我们介绍了html页面元素主要是通过标签和属性来进行定位 &#xff0c;只要满足唯一&#xff0c;无论是标签还是属性 &#xff0c;都能进行定位 。当然 &#xff0c;我们要通过selenium来进行定位 &#xff0c;同样还是…

富格林:利用正规方法提升出金收益

富格林悉知&#xff0c;黄金现货是众多黄金产品当中交易规则优势较为明显的存在&#xff0c;尤其是近段时间的行情走势十分强劲&#xff0c;这使得投资者们获得了更充足的做单机会。然而&#xff0c;要想在炒黄金的市场上取得成功&#xff0c;除了对市场有深刻的理解外&#xf…

总结:css中水平居中

css水平居中功能很常用&#xff0c;但一直没有系统的总结过&#xff0c;今天来总结水平下常见的方式&#xff1a; 1. 文本或内联元素的居中&#xff1a; 使用 text-align: center; 属性可以让内部的文本和内联元素在容器中水平居中。 .container {text-align: center; }2. 块…

小公司应该选择docker swarm还是k8s

对于小公司选择 Docker Swarm 还是 Kubernetes&#xff08;简称 K8s&#xff09;作为容器编排平台&#xff0c;应考虑以下几个关键因素&#xff1a; 技术复杂度与学习曲线&#xff1a; Docker Swarm&#xff1a;因其与 Docker 紧密集成&#xff0c;使用 Docker Compose 语法作…

MongoDB聚合运算符:$ne

文章目录 语法举例 $ne聚合运算符用于比较两个值是否不相等&#xff0c;如果他们的值不相等&#xff0c;返回true&#xff1b;如果他们的值相等&#xff0c;返回false。 语法 { $ne: [ <expression1>, <expression2> ] }$ne可以用来比较任何类型的值&#xff0c;…

【C++风云录】梦幻般的机器人世界:探索ROS、PCL、OpenCV和更多顶尖技术

机器人技术与计算机视觉世界&#xff1a;ROS、PCL、OpenCV等综合指南 前言 本文旨在深入探讨ROS&#xff08;机器人操作系统&#xff09;、PCL&#xff08;点云库&#xff09;、OpenCV&#xff08;开源计算机视觉库&#xff09;、Eigen&#xff08;线性代数库&#xff09;、B…

基于SpringBoot的在线五子连珠的设计与实现,前端采用vue框架;后端采用SpringBoot,mybatis

介绍 基于SpringBoot的在线五子连珠的设计与实现&#xff0c;主要是设计一款五子棋游戏&#xff0c;涉及登录注册的功能&#xff0c;人机对战、联机对战和积分排行榜的功能。其中人机对战中&#xff0c;电脑采用的是采用了一种基于局面分析的评分算法来确定机器人的下一步落子…