【与C++的邂逅】---- 函数重载与引用

在这里插入图片描述关注小庄 顿顿解馋(`▿´)
喜欢的小伙伴可以多多支持小庄的文章哦
📒 数据结构
📒 C++


引言 : 上一篇博客我们了解了C++入门语法的一部分,今天我们来了解函数重载,引用的技术,请放心食用 ~


文章目录

  • 一. 🏠 函数重载
    • 📒 函数重载的概念
    • 📒 函数重载的误区
      • 1.形参顺序不同是不同类型形参顺序不同
      • 2.函数返回值不是构成重载的条件
      • 3.函数在同一作用域才构成重载
      • 4.缺省参数不是构成重载的条件
    • 📒 C++支持函数重载的原因
  • 二. 🏠 引用
    • 📒 认识引用
    • 📒 引用特性
    • 📒 常引用
    • 📒 使用场景
    • 📒 传值 传引用 效率比较
    • 📒 引用的大小
    • 📒 引用和指针的区别

一. 🏠 函数重载

void Swap(int x ,int y)
{int tmp = 0;temp = x;x = y;y = temp;
}

这个函数想必不会陌生吧,可我们发现这个Swap函数只能交换整形,如果我同样调换这个函数就能交换浮点型呢?这里C++就给我们提供了函数重载的技术。

📒 函数重载的概念

函数重载:是函数的一种特殊情况,C++允许在同一作用域中声明几个功能类似的同名函数,这些同名函数的形参列表(参数个数或类型或类型顺序)不同,常用来处理实现功能类似数据类型不同的问题。

我们上代码来感受一下

形参类型不同

#include<iostream>
using namespace std;
// 1、参数类型不同
int Add(int left, int right)
{cout << "int Add(int left, int right)" << endl;return left + right;
}
double Add(double left, double right)
{cout << "double Add(double left, double right)" << endl;return left + right;
}

形参个数不同

void f()
{cout << "f()" << endl;
}
void f(int a)
{cout << "f(int a)" << endl;
}

形参顺序不同

void f(int a, char b)
{cout << "f(int a,char b)" << endl;
}
void f(char b, int a)
{cout << "f(char b, int a)" << endl;
}

只要满足一个条件即可发生重载

📒 函数重载的误区

1.形参顺序不同是不同类型形参顺序不同

void f(int a, int b)
{cout << "f(int a,char b)" << endl;
}
void f(int b,int a)
{cout << "f(int a,char b)" << endl;
}

这里形参顺序不同坑定能正常编译吧老铁,实则不然
在这里插入图片描述
注:函数重载形参顺序不同,指的是不同类型形参的顺序不同

2.函数返回值不是构成重载的条件

int func()
{cout << "use int func()" << endl;return 0;
}
void func()
{cout << "use void func(int a)" << endl;
}

对于这段代码同样不能正常编译,函数返回值不作为函数重载的条件,首先函数这里会产生调用歧义(语法上调哪一个都可以),同时函数返回值不一定要接收

3.函数在同一作用域才构成重载

namespace N1
{void Mul(int a, int b){cout << a * b << endl;}
}
namespace N2
{void Mul(int a, int b){cout << a * b * 2 << endl;}
}

此时这两个函数是否构成重载呢?
注:函数重载是要在同一作用域下的,这里N1和N2是两个不同的命名空间域
注:不同域可以定义同名,相同域也可以但要符合重载条件。

那如果我展开他们的命名空间呢?

using namespace N1;
using namespace N2;
Mul(1,2);
Mul(1,3);

此时这两个函数虽然都展开引入了全局域中,但仍然不构成重载,会产生调用歧义。

4.缺省参数不是构成重载的条件

void func(int a = 10)
{cout << "func(int a = 10)" << endl;
}
void func(int a = 2)
{cout << "func(int a = 2)" << endl;
}

能否构成函数重载呢?编译器会给我们答案
在这里插入图片描述
很显然,缺省参数是不构成函数重载的条件的

如果是这样呢?

void func()
{cout << "func(int a = 10)" << endl;
}
void func(int a = 2)
{cout << "func(int a = 2)" << endl;
}

此时两个函数确实构成重载,但会发生调用歧义。

📒 C++支持函数重载的原因

我们知道在C语言中函数不能同名,那为什么C++就支持重载,C语言不支持呢?我们得先来回顾一下编译和链接的过程

  • C语言

    编译链接的过程大致如下,可以参考博主之前写的文章编译与链接
    在这里插入图片描述
    在链接时,我们会进行符号决议和重定位,也就是我们调用函数时,编译器会根据函数名符号表中的符号去找函数地址,与我们.c文件调用的符号链接起来

补充:

  1. 在只有函数声明的文件中,在编译过程中没有函数的地址,但能通过语法检查
  2. 有函数定义才能形成一系列的汇编指令,函数定义的第一条就是函数的地址

总结:C语言直接通过函数名字去查找函数,这样无法区分,故不支持重栽

  • C++
    与C语言不同的是,C++在链接时是通过修饰后的函数名去查找,可以起到区分的作用,因此支持重载。

具体是怎么重载的呢?我们上图

//g++编译器 Linux环境
void f(int a,char b);
void f(char a,int b);

在这里插入图片描述
在这里插入图片描述
我们可以发现由于形参列表的不同(c表示char i表示int),构成了修饰名的不同,编译器将函数参数类型信息添加到原函数名后

小补充:在不同的平台,函数名的修饰规则是不同的。

windows系统
在这里插入图片描述

对比Linux会发现,windows下vs编译器对函数名字修饰规则相对复杂难懂,但道理都是类似的,我们就不做细致的研究了。

二. 🏠 引用

📒 认识引用

  • 引用的概念

引用不是新定义一个变量,而是给已存在变量取了一个别名,编译器不会为引用变量开辟内存空间,它和它引用的变量共用同一块内存空间。

  • 引用的语法形式
类型& 引用变量名(对象名) = 引用实体;// 注意引用类型要和它的引用实体的类型相同
int a = 10;
int& b = a;
cout << &a << endl;
cout << &b << endl;
int* p = &a;
int* & =  p;
double d = 1.0;
double& pd = d;

输出
008FF838
008FF838 // 引用和变量确实共用一块空间

📒 引用特性

  • 引用定义时必须初始化
int x = 0;
int& a = x;
  • 一个变量可以有多个引用
int x = 0;
int & a = x;
int & b = x;
int & c = b;
c++;//这里c改变a b x都会改变
  • 引用一旦初始化,不可改变引用实体
int a = 0;
int b = 1;
int& pa = a;
pa = b ; //这里是把b的值赋给a/pa;

📒 常引用

权限的平移

int x = 0;
int &y = x ;
//引用
const int m = 10;//此时m是只读
const int & pm = m;
//指针
const int* p1 = &m;
const int* p2 = p1;

权限的放大

//引用
const int m = 10;
int& r = m; //此时m只能读取,int&是可读可写 权限放大是不行的//指针
const int* p1 = &m;
int* p2 = p1;//p1只读 权限放大不可以//普通变量赋值的拷贝
in p = m; //此时是把m的值拷贝给p,p的修改不影响m

权限的缩小

int x = 0;
//引用
const int& z = x;
//z++不行 因为只能读
//指针
int* p3 = &x;
const int* p4 = p3;
double d = 1.9;
//int& t = d; 会报错
const int& r = d;int x = 1,y = 0;
const int& r = x + y;
//int& pr = x + y; 会报错

在类型转换和表达式求值时,会产生临时变量(因为要存储他们运算后的结果),而临时变量具有常性(相当于被const修饰,只读),这里用int&接收造成权限的放大。

总结:对于指针和引用权限可以平行缩小,但不能放大;普通变量赋值没有权限之说。

📒 使用场景

  • 作为函数的形参
void Swap(int& left, int& right)
{int temp = left;left = right;right = temp;
}
  • 作为返回值
int& Count()
{static int n = 0;n++;// ...return n;
}

📒 传值 传引用 效率比较

#include <time.h>
struct A{ int a[10000]; };
void TestFunc1(A a){}
void TestFunc2(A& a){}
void TestRefAndValue()
{A a;// 以值作为函数参数size_t begin1 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc1(a);size_t end1 = clock();// 以引用作为函数参数size_t begin2 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc2(a);size_t end2 = clock();
// 分别计算两个函数运行结束后的时间cout << "TestFunc1(A)-time:" << end1 - begin1 << endl;cout << "TestFunc2(A&)-time:" << end2 - begin2 << endl;
}

输出结果
“TestFunc1(A)-time:” :27
“TestFunc2(A&)-time:”:0

总结:传引用比传值效率高出很多,可以认为在语法层面上传引用几乎没开空间

那是否引用真的没开空间呢?

int x = 2;
int* p = &x;
int& pr = x;

在这里插入图片描述
我们转到反汇编观察发现,定义引用时其实也是把变量的地址放到引用变量里。
换句话说,引用的底层也是指针,基于这个理解,我们来看下面的问题。

📒 引用的大小

#include<iostream>
using namespace std;
//x64
cout << sizeof(int&) << endl;
cout << sizeof(short&) << endl;
cout << sizeof(long&) << endl;
//x86
cout << sizeof(int&) << endl;
cout << sizeof(short&) << endl;
cout << sizeof(long&) << endl;

输出结果:
x64环境下
4
2
4
x86环境下
4
2
4

总结:引用大小在语法层面上规定是它引用实体类型大小,毕竟引用是一种语法,sizeof没有意义

那如果是这样呢?

#include<iostream>
using namespace std;
struct Test
{
int& age;
}
struct Test t;
cout << sizeof(t)<<endl;

结合我们之前学的结构体内存对齐知识,这里输出结果是否应该等于4?

输出结果:
32位环境下
4
64位环境下
8
//你是否感到疑惑?同时我这里为什么要以环境来区分?

实际上,结合我们之前的结论引用底层实质是个地址,当我们用结构体定义出一个实际的对象时,底层就有了蓝图,那就需要去翻译和识别它是个指针类型了。

📒 引用和指针的区别

指针引用
指针存的是变量的地址引用是变量的别名
有空指针NULL没有空引用
有多级指针没有多级引用
指针可以改变指向引用初始化后不可改变引用实体
指针相对安全性低引用相对安全性高
sizeof(指针)始终是地址空间所占字节大小引用大小为引用实体类型的大小
自增是向后偏移一个类型的大小自增是引用实体增加
指针访问实体要解引用引用访问实体编译器自己处理
指针不一定要初始化引用一定要初始化

学到知识的小伙伴,不妨给小庄一个三连呀 ~

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

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

相关文章

【JavaWeb】Day31.SpringBootWeb请求响应——分层解耦(二)

3.IOC&DI 3.1 IOC&DI入门 完成Controller层、Service层、Dao层的代码解耦 思路&#xff1a; 1. 删除Controller层、Service层中new对象的代码 2. Service层及Dao层的实现类&#xff0c;交给IOC容器管理 3. 为Controller及Service注入运行时依赖的对象 Controller程序…

隐私计算实训营第六讲-隐语PIR介绍及开发实践

隐私计算实训营第六讲-隐语PIR介绍及开发实践 文章目录 隐私计算实训营第六讲-隐语PIR介绍及开发实践1.隐语实现PIR总体介绍1.1按服务器数量分类1.2按查询类型分类 2. Index PIR - SealPIR3. Keyword PIR - Labeled PSI4.隐语PIR功能分层5.隐语PIR后续计划PIR协议开发PIR调用框…

JavaScript 数组元素交互最优解

利用 ES6 解构赋值&#xff1a; let arr [1, 2, 3, 4, 5];// 交互下标 1,4 元素的值 [arr[1], arr[4]] [arr[4], arr[1]];// 输出&#xff1a; [1, 5, 3, 4, 2] console.log(arr);浏览器控制台效果&#xff1a;

Node.js-知识点学习总结归纳

Node.js-知识点学习总结归纳 安装nodenode运行方式通过Node.js直接运行js文件&#xff08;也就不用通过网页html了&#xff09;绝对路径调用:相对路径调用&#xff1a;直接运行js命令&#xff1a; Vscode控制台使用node运行js文件 安装node 这个就不用讲了吧&#xff0c;网上搜…

硬件知识:点亮led

怎么做 1、看原理图&#xff0c;去顶控制LED的引脚 1.1led 1.2gpio 1、使能gpio模块&#xff0c;设置Power/clock control,向gpio模块提供电源和时钟 2.设置引脚模式为gpio模式 3.设置方向output/input4、读取引脚&#xff0c;获得当前电频到底是低电平/高电平&#xff1b;这…

数据库:Redis数据库

一、非关系型数据库 1.什么是非关系型数据库 非关系型数据库&#xff08;Non-relational Database&#xff09;又称NoSQL数据库是一种不同于传统关系型数据库管理系统&#xff08;RDBMS&#xff09;的数据存储解决方案。NoSQL这个术语最初意味着"Not Only SQL"&…

1999-2022年上市公司员工人数数据

1999-2022年上市公司员工人数数据 1、时间&#xff1a;1999-2022年 2、指标&#xff1a;证券代码、时间、员工人数 3、来源&#xff1a;整理自csmar 4、范围&#xff1a;上市公司 5、指标解释&#xff1a; 上市公司员工人数是衡量公司规模和发展状的重要指标。该数据直接…

阿赵UE学习笔记——24、动画播放控制

阿赵UE学习笔记目录   大家好&#xff0c;我是阿赵。   继续学习虚幻引擎的使用。关于UE的动画系统&#xff0c;之前学习了很多&#xff0c;包括动画合成或者动画蒙太奇等&#xff0c;实际上最后得到的都是一个动画片段。那么这些动画片段&#xff0c;是需要怎样播放控制呢…

Taro多行文本最多展示5行,超出“查看更多”展示,点击弹层

Taro中&#xff0c;页面需求&#xff1a; 多行文本&#xff0c;展示最多展示5行&#xff0c;超出5行&#xff0c;展示“查看更多”按钮&#xff0c;点击弹层展示文本详细信息。 弹层代码就不说了&#xff0c;着重说一下怎么获取区域高度&#xff5e; 1.区域设置max-height&am…

ArcGIS Pro打不开Excel?Microsoft驱动程序安装不上?

刚用ArcGIS pro的朋友们可能经常在打开xls或者xlsx文件的时候都会提示&#xff0c;未安装所需的Microsoft驱动程序。 怎么办呢&#xff1f;当然&#xff0c;按照提示装一下驱动就会好吗&#xff1f;有什么状况会出现&#xff1f;有什么临时替代方案呢&#xff1f; 全文目录&a…

为什么要选择第三方软件测试机构?CMA、CNAS第三方软件测试机构推荐

第三方软件测试机构是独立于软件开发方和软件使用方的中立机构&#xff0c;致力于对软件产品进行全面、客观、专业的测试和评估&#xff0c;为软件开发方和使用方提供全面的技术支持和服务。 一、为什么要选择第三方软件测试机构   1、专业性强&#xff1a;拥有专业的测试团…

基本电路理论-电流和电压的参考方向

&#x1f308;个人主页&#xff1a;会编程的果子君 &#x1f4ab;个人格言:“成为自己未来的主人~” 电流及参考方向 电流&#xff1a;带电粒子有规则的定向移动 电流强度&#xff1a;单位时间内通过导体横截面的电荷量&#xff0c;即&#xff1a;idq/dt 单位&#xff1a…

Vue3_2024_7天【回顾上篇watch常见的后两种场景】完

随笔&#xff1a;这年头工作不好找咯&#xff0c;大家有学历提升的赶快了&#xff0c;还有外出人多注意身体&#xff0c;没错我在深圳这边阳了&#xff0c;真的绝啊&#xff0c;最尴尬的还给朋友传染了&#xff01;&#xff01;&#xff01; 之前三种的监听情况&#xff0c;监听…

【Qt 学习笔记】Qt 背景介绍

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ Qt 背景介绍 文章编号&#xff1a;Qt 学习笔记 / 01 文章目录 Qt 背景…

CSS面试题---基础

1、css选择器及优先级 选择器优先级&#xff1a;内联样式>id选择器>类选择器、属性选择器、伪类选择器>标签选择器、微元素选择器 注意&#xff1a; !important优先级最高&#xff1b; 如果优先级相同&#xff0c;则最后出现的样式生效&#xff1b; 继承得到的样式优先…

Sora可能会改变我们的思维方式

当经济学家评估生成式人工智能对人类工作的影响时&#xff0c;教育家也有自己的担忧&#xff0c;毕竟在写作、答疑等领域&#xff0c;AI所展现的能力已经令许多人类望尘莫及&#xff0c;学者们可能有这样的思考&#xff1a;散文是否会继续主导人类的交流和论证。 社交媒体和新闻…

基于Scala开发Spark ML的ALS推荐模型实战

推荐系统&#xff0c;广泛应用到电商&#xff0c;营销行业。本文通过Scala&#xff0c;开发Spark ML的ALS算法训练推荐模型&#xff0c;用于电影评分预测推荐。 算法简介 ALS算法是Spark ML中实现协同过滤的矩阵分解方法。 ALS&#xff0c;即交替最小二乘法&#xff08;Alte…

4.2 JavaWeb Day05分层解耦

三层架构功能 controller层接收请求&#xff0c;响应数据&#xff0c;层内调用了service层的方法&#xff0c;service层仅负责业务逻辑处理&#xff0c;其中要获取数据&#xff0c;就要去调用dao层&#xff0c;由dao层进行数据访问操作去查询数据&#xff08;进行增删改查&…

new mars3d.layer.HeatLayer({实现动态修改热力图半径

1.使用热力图插件的时候&#xff0c;实现动态修改热力图效果半径 2.直接修改是不可以的&#xff0c;因为这个是热力图本身的参数。 因此我们需要拿到这个热力图对象之后&#xff0c;参考api文档&#xff0c;对整个 heatLayer.heatStyle进行传参修改。 heatStyle地址&#x…

HarmonyOS 应用开发之featureAbility接口切换particleAbility接口切换

featureAbility接口切换 FA模型接口Stage模型接口对应d.ts文件Stage模型对应接口getWant(callback: AsyncCallback<Want>): void; getWant(): Promise<Want>;ohos.app.ability.UIAbility.d.tslaunchWant: Want;startAbility(parameter: StartAbilityParameter, c…