c++三大概念要分清--重载,隐藏(重定义),覆盖(重写)

目 录

在这里插入图片描述

一、重载

**(1)概念:**在同一个作用域内;函数名相同,参数列表不同(参数个数不同,或者参数类型不同,或者参数个数和参数类型都不同),返回值类型可相同也可不同;这种情况叫做c++的重载!
(2)特征:
在同一作用域(例如在同一个类中);
函数名字相同;
参数列表不同。
(3)重载达到的效果:
调用函数名相同的函数,会根据实参的类型和实参顺序以及实参个数选择相应的函数。C++ 函数重载是一种静态多态(又叫做静态联编,静态绑定)。
(4)举例:

#include <stdio.h>
#include <iostream> using namespace std;int Add(int a, int b)
{return a + b;
}float Add(float a, float b)
{return a + b;
}int main()
{cout << Add(4, 5) << endl; // 调用 int Add(int a,int b)cout << Add(2.5f, 3.7f) << endl; // 调用 float Add(float a,float b)return 0;
}/*
输出结果:9
6.2
*/

此时,两个函数Add();在同一作用域,函数名相同都是 Add,参数类型不同,就构成了 C++ 中的函数重载。

二、覆盖(又叫重写)

(1)概念:
当在子类中定义了一个与父类完全相同的虚函数时,则称子类的这个函数重写(也称覆盖)了父类的这个虚函数。

(2)覆盖:

不同的作用域(分别位于派生类与基类);
函数名字相同;
参数相同;
基类函数必须有virtual关键字。
(3)什么是在子类中定义了一个与父类完全相同的虚函数:
有两种情况:

  • 就是说子类中的虚函数和父类中的虚函数,函数名,参数个数,参数类型,返回值类型都相同;这种情况下子类的这个虚函数重写的父类中的虚函数,构成了重写。
  • 协变—是说子类中的虚函数和父类中的虚函数,函数名,参数个数,参数类型都相同,只是返回值类型不同;父类的虚函数返回父类的指针或者引用,子类虚函数返回子类的指针或者引用;这种情况下子类的这个虚函数也重写了父类中的虚函数,也构成了重写;——我们把这种特殊的情况叫做协变。

(4)覆盖(重写)达到的效果:

  • 在子类中重写了父类的虚函数,那么子类对象调用该重写函数,调用到的是子类内部重写的虚函数,而并不是从父类继承下来的虚函数;(这其实就是动态多态的实现);
  • 在子类中重写了父类的虚函数,如果用一个父类的指针(或引用)指向(或引用)子类对象,那么这个父类的指针或引用将调用该子类重写的虚函数;相反,如果用一个父类的指针(或引用)指向(或引用)父类的对象,那么这个父类的指针(或引用)将调用父类的虚函数。

(5)举例一:普通重写

// 普通重写+函数重载
#include <stdio.h>
#include<iostream> 
using namespace std;class Base
{
public:virtual void Print() // 父类虚函数{printf("This is Class Base!\n");}
};class Derived : public Base
{
public:void Print() // 子类1虚函数,重写了父类的虚函数{printf("This is Class Derived!\n");}
};int main()
{Base Cbase; // 父类对象Derived Cderived; // 子类对象Cbase.Print();Cderived.Print();cout << "---------------" << endl;Base *p1 = &Cderived; // 父类指针指向子类对象 Base *p2 = &Cbase; // 父类指针指向父类对象	p1->Print(); p2->Print();
}/*
输出结果:This is Class Base!
This is Class Derived!
---------------
This is Class Derived!
This is Class Base!
*/

(6)举例二:协变重写

// (协变)重写+函数重载
#include <stdio.h>
#include<iostream> 
using namespace std;class Base
{
public:virtual Base &Print() // 父类虚函数{printf("This is Class Base!\n");return *this;}
};class Derived :public Base
{
public:Derived &Print() // 子类虚函数,重写了父类的虚函数{printf("This is Class Derived!\n");return *this;}
};int main()
{Base Cbase;Derived Cderived;Cbase.Print();Cderived.Print();cout << "---------------" << endl;Base *p1 = &Cderived;Base *p2 = &Cbase;p1->Print();p2->Print();
}/*
输出结果:This is Class Base!
This is Class Derived!
---------------
This is Class Derived!
This is Class Base!
*/

(7)重载与覆盖(重写)有什么区别?

  • 范围区别:重载的函数在同一类中,而重写的函数在不同的类(基类和派生类)中;
  • 参数列表区别:重载要求参数列表不同,而覆盖要求参数列表相同;
  • virtual 的区别:重载函数和被重载函数可以被 virtual 修饰,也可以没有,而重写的基类函数必须要 有virtual 修饰;
  • 调用方法不同:重载是根据调用时的实参列表来选择方法体的,而覆盖是根据对象的类型来决定的。

三、隐藏(重定义)

(1)概念:是指派生类的函数屏蔽了与其同名的基类函数,规则如下:

  • 如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有 virtual 关键字,此时基类的函数才被隐藏。
  • 如果派生类的函数与基类的函数同名,但是参数不同,则不论有无 virtual 关键字, 基类的函数都将被隐藏。
    在调用一个类的成员函数时,编译器会沿着类的继承链逐级地向上查找函数的定义,如果找到了就停止查找了。所以,如果一个派生类和一个基类都存在同名(暂且不论参数是否相同)的函数,而编译器最终选择了在派生类中的函数,那么就说这个派生类的成员函数“隐藏”了基类的成员函数,也就是说它阻止了编译器继续向上查找函数的定义。
    (2)特征:
  • 必须分别位于派生类和基类中;
  • 必须同名;
  • 参数不同的时候本身已经不构成覆盖关系了,所以此时是否是 virtual 函数已经不重要了。

(3)隐藏(重定义)的使用范围:

隐藏的不光是类的成员函数,还可以是类的成员变量;

(4)隐藏(重定义)的直接效果:

如果在父类和子类中有相同名字的成员,那么在子类中,会将父类的成员隐藏;隐藏以后的直接效果就是:无论在子类的内部或者外部(通过子类成员)访问该成员;全都是访问子类的同名成员。
如果用一个父类的指针(或引用)指向(或引用)子类对象,那么这个父类的指针或引用将调用父类的同名函数;
(5)举例:

#include<iostream> 
using namespace std;class Base
{
public:Base(int x = 1) :value(x) {}void Print1() // 父类函数不是虚函数{cout << "Base Print1():" << value << endl;}virtual void printf2() // 父类函数是虚函数{cout << "Base Print2():" << value << endl;}int value;
};class Derived :public Base
{
public:Derived(int x = 2) :value(x) {}void Print1() // 函数同名,且参数相同,同时父类函数不是虚函数,构成“隐藏”{cout << "Derived Print1():" << value << endl;}void Print2(int a) // 函数同名,但是参数不同,无论父类函数是不是虚函数,都构成“隐藏”{cout << "Derived Print2():" << value << endl;a = 0;}int value; // 子类成员数据,隐藏了子类的同名数据
};int main()
{Derived Cderived;cout << Cderived.value << endl; // 调用子类的成员数据Cderived.Print1(); // 隐藏Cderived.Print2(1); // 隐藏Base &Cbase = Cderived; // 父类的指针(或引用)指向(或引用)子类对象Cbase.Print1(); // 将调用父类的同名函数(隐藏)
}/*
输出结果:2
Derived Print1():2
Derived Print2():2
Base Print1():1
*/

(6)隐藏(重定义)与覆盖(重写)有什么区别?

  • virtual 的区别:隐藏的基类函数不需要用 virtual 修饰,而覆盖的基类函数必须要有 virtual 修饰;
  • 参数列表区别:隐藏的基类和派生类的函数参数列表可以不同,而覆盖的基类和派生类的函数参数列表必须相同。

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

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

相关文章

uniapp项目实践总结(二十五)苹果 ios 平台 APP 打包教程

导语:当你的应用程序开发完成后,在上架 ios 应用商店之前,需要进行打包操作,下面就简单介绍一下打包方法。 目录 准备工作注册账号生成证书打包配置准备工作 在打包之前,请保证你的 uniapp 应用程序编译到 ios 模拟器或者是真机调试基座环境下是可以正常运行的,苹果打包…

input输入表头保存excel文件

input输入表头 input输入表头 &#xff08;input内除了/&#xff0c;空格 回车 标点符号等 全部作为单元格分隔符&#xff09;保存/storage/emulated/0/代码文件/ 没有就创建文件名命名方法&#xff1a;编号. 库 时间戳嗨&#xff01;听说你有个需求&#xff0c;想根据用户输入…

CSS基础语法第二天

目录 一、复合选择器 1.1 后代选择器 1.2 子代选择器 1.3 并集选择器 1.4 交集选择器 1.4.1超链接伪类 二、CSS特性 2.1 继承性 2.2 层叠性 2.3 优先级 基础选择器 复合选择器-叠加 三、Emmet 写法 3.1HTML标签 3.2CSS 四、背景属性 4.1 背景图 4.2 平铺方式 …

使用sqlmap的 ua注入

文章目录 1.使用sqlmap自带UA头的检测2.使用sqlmap随机提供的UA头3.使用自己写的UA头4.调整level检测 测试环境&#xff1a;bWAPP SQL Injection - Stored (User-Agent) 1.使用sqlmap自带UA头的检测 python sqlmap.py -u http://127.0.0.1:9004/sqli_17.php --cookie“BEEFHOO…

01-Zookeeper特性与节点数据类型详解

上一篇&#xff1a; 在了解Zookeeper之前&#xff0c;需要对分布式相关知识有一定了解&#xff0c;什么是分布式系统呢&#xff1f;通常情况下&#xff0c;单个物理节点很容易达到性能&#xff0c;计算或者容量的瓶颈&#xff0c;所以这个时候就需要多个物理节点来共同完成某项…

C++ 重载运算符和重载函数

C 允许在同一作用域中的某个函数和运算符指定多个定义&#xff0c;分别称为函数重载和运算符重载。 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明&#xff0c;但是它们的参数列表和定义&#xff08;实现&#xff09;不相同。 当您调用一个重…

【论文极速读】Prompt Tuning——一种高效的LLM模型下游任务适配方式

【论文极速读】Prompt Tuning——一种高效的LLM模型下游任务适配方式 FesianXu 20230928 at Baidu Search Team 前言 Prompt Tuning是一种PEFT方法&#xff08;Parameter-Efficient FineTune&#xff09;&#xff0c;旨在以高效的方式对LLM模型进行下游任务适配&#xff0c;本…

基于大语言模型扬长避短架构服务

秘诀&#xff1a; 扬泛化之长&#xff0c; 避时延之短

CocosCreator3.8研究笔记(二十四)CocosCreator 动画系统-动画编辑器实操-关键帧实现动态水印动画效果

上一篇&#xff0c;我们介绍了动画编辑器相关功能面板说明&#xff0c;感兴趣的朋友可以前往阅读&#xff1a; CocosCreator3.8研究笔记&#xff08;二十三&#xff09;CocosCreator 动画系统-动画编辑器相关功能面板说明。 熟悉了动画编辑器的基础操作&#xff0c;那么再使用动…

基于Javaweb的护肤品推荐系统 /基于ssm的护肤品销售系统

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&a…

八大排序(三)堆排序,计数排序,归并排序

一、堆排序 什么是堆排序&#xff1a;堆排序&#xff08;Heap Sort&#xff09;就是对直接选择排序的一种改进。此话怎讲呢&#xff1f;直接选择排序在待排序的n个数中进行n-1次比较选出最大或者最小的&#xff0c;但是在选出最大或者最小的数后&#xff0c;并没有对原来的序列…

毕业设计选题uniapp+springboot新闻资讯小程序源码 开题 lw 调试

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人七年开发经验&#xff0c;擅长Java、Python、PHP、.NET、微信小程序、爬虫、大数据等&#xff0c;大家有这一块的问题可以一起交流&#xff01; &#x1f495;&…

基于 QT 实现一个 Ikun 专属桌面宠物

Step0、实现思路 想到的思路有两种&#xff1a; 1、使用 QT 的状态机模式&#xff0c;参考官网文档&#xff0c;这个模式的解耦最佳 2、使用原生 Wigets&#xff0c;将窗口设置为透明无框&#xff0c;循环播放桌面宠物的状态 本文采用第二种思路&#xff0c;实现一个极简版…

低功耗引擎Cliptrix为什么可以成为IOT的高效能工具

在万物互联的时代&#xff0c;现代人已普遍接受电视、音箱等电器设备具备智能化能力&#xff0c;也是在这个趋势下&#xff0c;我们身边越来越多的iOT设备联网和交互成为刚需。 但iot设备也面临到一些非常显著的痛点&#xff0c;例如iot设备的内存、处理器等核心元件无法与手机…

[Java·算法·困难]LeetCode124.二叉树中的最大路径和

每天一题&#xff0c;防止痴呆 题目示例分析思路1题解1 &#x1f449;️ 力扣原文 题目 二叉树中的 路径 被定义为一条节点序列&#xff0c;序列中每对相邻节点之间都存在一条边。同一个节点在一条路径序列中 至多出现一次 。该路径 至少包含一个 节点&#xff0c;且不一定经…

对pyside6中的textedit进行自定义,实现按回车可以触发事件。

以下方法不算最优解。因为这个ui文件很容易重新编译&#xff0c;使写在ui.py里面的代码被删掉。 所以更好的方法应该是在主代码当中单独定义控件。并且使用布局添加控件到界面中。 以下内容纯为旧版实现&#xff0c;仅供参考&#xff1a; 我的实现方法是&#xff0c;先用qt de…

拒绝水文!八大排序(三)【适合初学者】快速排序

文章目录 快速排序递归实现霍尔法优化 挖坑法前后指针法 快速排序非递归 大家好&#xff0c;我是纪宁&#xff0c;这篇文章将向大家介绍非常有名气的一款排序&#xff1a;快速排序 回忆到我们刚开始学习C语言的时候。经常会使用到一个库函数&#xff1a; qsort函数 &#xff0…

【JavaEE】JavaScript

JavaScript 文章目录 JavaScript组成书写方式行内式内嵌式外部式&#xff08;推荐写法&#xff09; 输入输出变量创建动态类型基本数据类型数字类型特殊数字值 String转义字符求长度字符串拼接布尔类型undefined未定义数据类型null 运算符条件语句if语句三元表达式switch 循环语…

百元价位开放式耳机哪款好、百元耳机性价比最高的开放式耳机推荐

现如今越来越多的人选择开放式耳机&#xff0c;因为开放式耳机具有多重优点&#xff0c;首先是舒适性。由于它们不需要插入耳道&#xff0c;长时间佩戴也不会产生不适感。其次&#xff0c;开放式耳机在保持与外界的连接上表现出色&#xff0c;使得户外活动更加安全。另外&#…

Unity Game FrameWork—模块使用—资源热更新

工程选项配置​ json文件解析不要用默认的unity解析方式&#xff0c;unity解析有问题&#xff0c;使用StarForce.LitJsonHelper​ ​ 资源模式改为热更新模式​ ​ 配置文件配置​ BuiltinDataComponent自定义组件引用率了Buildinfo.txt配置文件&#xff0c;该文件配置了热更…