【C++对于C语言的扩充】函数重载、引用以及内联函数

请添加图片描述

文章目录

  • 🚀前言
  • 🚀函数重载
    • 注意:
    • ✈️为什么C++可以实现函数重载,而C语言却不行呢?
  • 🚀引用
    • ✈️引用的特性
    • ✈️C++中为什么要引入引用
    • ✈️引用与指针的区别
  • 🚀内联函数
    • ✈️内联函数特性

🚀前言

大家好啊!好久没更文了,课多还有就是备战前几天考完的蓝桥杯,好了不多bb,接着带大家从C语言过度到C++!!!

🚀函数重载

当年本贾尼,本大爷在编C++的时候觉得C语言中的函数不够方便,于是就整了个函数重载

函数重载:在C++中,允许同一作用域声明几个功能类似的同名函数,这些函数具有相同的函数名,但具有不同的参数列表(形参数不同、类型不同 以及类型顺序不同)常用来处理实现功能类似数据类型不同的问题。

实话说,干巴巴的文字没一点意思,还是给铁子们来点🌰:

  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;
}
int main()
{Add(10, 20);Add(10.1, 20.2);return 0;
}

运行输出:

int Add(int left, int right)
double Add(double left, double right)

上面这两个个add函数如果用C语言编写还不得整个add_intadd_double啊,这多麻烦,C++的函数重载简直不要太爽

  1. 形参数不同
void f(int a){cout<<"void f(int a)"<<endl;
}
void f(int a,int b){cout<<"void f(int a,int b)"<<endl;
}
int main(){f(1);f(2,3);return 0;
}

运行输出:

void f(int a)
void f(int a,int b)
  1. 形参类型顺序不同
void f(int a,double b){cout<<"void f(int a,double b)"<<endl;
}
void f(double a,int b){cout<<"void f(double a,int b)"<<endl;
}
int main(){f(1,2.0);f(1.0,2);return 0;
}

运行输出:

void f(int a,double b)
void f(double a,int b)

注意:

函数的返回类型不同并不构成函数重载
也就是:

void add(int a,int b){....}
int add(int a,int b){return 0}

上面的add函数并不构成重载

为什么呢?我们接着往下看⬇️

✈️为什么C++可以实现函数重载,而C语言却不行呢?

这就与编译器有关了,一般C/C++的程序运行起来可以分为两个大的过程——编译和链接,在程序编译过程中会将每一个函数解析成唯一的标识符,然后将函数标识符及其地址编成符号表(符号表,我们可以简单把它理解成是一张维护标识符(变量、常量、函数)以及其相关信息的映射表),而后的链接过程中遇到一个函数调用时,就会通过该函数的标识符在符号表中找到对应的函数地址并与函数调用关联起来。

而C++可以实现实现重载,C却不能,主要是因为函数在编译过程解析成标识符这一步不同
比如:

void add(int a,int b){}
上面的函数在C编译器上一般会被编译类似 add 这样的标识符
而在C++中则会编译成类似 addii 这种标识符,后面的两个i表示该函数有两个int类型的形参

就是因为C语言在处理函数标识符时只与函数名有关,而一旦函数名相同就造成函数重定义,所以就注定C语言无法构成重载,而C/C++对于函数标识符的处理都与返回值无关所以返回值不同无法构成重载

🚀引用

引用:语法层面上就是已定义变量的别名,与该变量共用一块内存空间
栗子🌰:
类型& 引用变量名(对象名) = 引用实体
引用的类型与被引用的变量类型一致

int a = 10;
int& b = a;
b = 5;
b 对于a的关系 就相当于你的名字和小名以及外号的关系 都指的是你
对b进行修改其实就相当于改了a

其实不妨这么理解,定义a就相当于想系统申请了一块空间名字是a,而引用b则是给这块空间又起了一个名字,如果再次int& c = b就相当于这块空间又有了一个名字c,对a,b,c操作实际上都是对这块空间操作,所以a,b,c的值都会更改

✈️引用的特性

  • 引用必须初始化:
    这很好理解,引用就是所定义变量的别名,如果不初始化那么引用就没有所代表的内存空间
int main(){int& a;return 0;
}

不初始化就会报错
在这里插入图片描述

  • 引用一旦引用实体后,就不能引用别的变量了
int main()
{int a = 5;int c = 1;int& b=a;b = c;//这样b就是c的别名了吗?cout << a << " " << b << " " << c << endl;return 0;
}

运行输出:

1 1 1

其中b = c并非是b成了c的引用,而是将c的值赋给b,而ba的引用,b改了,a也就改了

✈️C++中为什么要引入引用

C/C++都是追求效率的语言,而C++中引入的引用其实就是代替了C中指针的大部分职能,比如在函数传参过程中如果是传值调用,这样就会出现拷贝,这极大的降低了效率,而C语言中通常会通过指针使用传址调用提高效率,而C++中则可以使用传引用做到与C中指针的传址调用同样的效果,甚至更便捷、安全。

栗子🌰:

//通过传引用交换变量
void swap(int& a,int& b){int tmp = a;a = b;b = tmp;
}
int main(){int a = 0,b = 4;swap(a,b);return 0;
}

注意⚠️:引用可以作为函数的返回值,一旦引用对象的生命周期只在函数内则会造成,返回的引用属于悬空引用,即引用指向的空间已销毁

✈️引用与指针的区别

大家应该应该注意到上文对于引用,我只说引用在语法上是已定义变量的别名,与该变量共用一块内存空间,其实在底层实现上引用本质就是指针

注意: 引用只能完成指针较为简单的部分,但是不能替代指针!!

  1. 引用概念上定义一个变量的别名,指针存储一个变量地址
  2. 引用在定义时必须初始化,指针没有要求
  3. 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型实体
  4. 没有NULL引用,但有NULL指针
  5. 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占4个字节)
  6. 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小
  7. 有多级指针,但是没有多级引用
  8. 访问实体方式不同,指针需要显式解引用,引用编译器自己处理(编译器在底层帮我们创建指针)
  9. 引用比指针使用起来相对更安全(因为引用必须初始化,而指针不用,没有空引用,但是有空指针)
  10. 在有些需要用二级指针的场景,较难理解,用引用就可以简化一点

🚀内联函数

内联函数其实是本大爷为了解决C中那让人恶心的宏函数而设计出来的

宏缺点:

1、不能调试(预处理阶段宏就被处理了)
2、没有类型安全的检查
3、有些场景下非常复杂,容易出错,不容易掌握

内联函数:以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数调用建立栈帧的开销,内联函数提升程序运行的效率
栗子🌰:

inline int add(int a,int b){return a + b;
}
int main(){int a = add(1,2);return 0;
}

✈️内联函数特性

  1. inline是一种以空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会用函数体替换函数调用

缺陷:由于是在调用处展开,则代码量将会扩大,也就导致目标文件的增大.
优势:少了调用开辟函数栈帧的开销,提高程序运行效率

  1. inline对于编译器而言只是一个建议(所以内不内联得看编译器脸色),不同编译器关于inline实现机制可能不同,一般建议:将函数规模较小(即函数不是很长,具体没有准确的说法,取决于编译器内部实现不是递归、频繁调用的函数采用inline修饰,否则编译器将不会采用=内联方式
  2. inline不建议声明和定义分离,分离会导致链接错误。因为inline被展开,就没有函数地址了,链接就会找不到,也就是定义和声明在一块,因为如果不在一块链接过程就会出问题,因为内联后的函数被编译进符号表,遇到函数调用时就会出问题
    栗子🌰:
//定义和声明在一块
inline int add(int a,int b){return a + b;
}

今天的分享到这里就结束啦!如果觉得文章还不错的话,可以三连支持一下,您的支持就是阿辉前进的动力!

请添加图片描述

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

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

相关文章

策略模式(Strategy Pattern)在JAVA中的应用

设计模式是软件工程中的一套被广泛认可的解决特定问题的最佳实践。它们是在多年的软件开发实践中总结出的有效方法。策略模式是JAVA中常用的一种行为型设计模式&#xff0c;它定义了一系列算法&#xff0c;并将每一个算法封装起来&#xff0c;使它们可以互换&#xff0c;让算法…

GitHub提交PR

本教程只做开源代码库Github工程提交pr的教程&#xff0c;不做其他的深入的讲解 Github和Gitlab的操作类似&#xff0c;只不过Github叫PR&#xff0c;GitLab叫MR&#xff0c;基本上做法是一致的 以开源项目QuickChat为例 https://github.com/Binx98/QuickChat https://github…

C++项目 -- 负载均衡OJ(一)comm

C项目 – 负载均衡OJ&#xff08;一&#xff09;comm 文章目录 C项目 -- 负载均衡OJ&#xff08;一&#xff09;comm一、项目宏观结构1.项目功能2.项目结构 二、comm公共模块1.util.hpp2.log.hpp 一、项目宏观结构 1.项目功能 本项目的功能为一个在线的OJ&#xff0c;实现类似…

研发岗-统信UOS系统配置npm git等前端常用配置

第一步 获取root权限 配置环境等都需要用到root权限&#xff0c;所以我们先获取到root权限&#xff0c;方便下面的操作 下载软件 在UOS应用商店下载的所需应用 版本都比较低 安装node 官网下载了【arm64】的包&#xff0c;解压到指定文件夹&#xff0c;设置链接&#xff0…

苹果电脑启动磁盘是什么意思 苹果电脑磁盘清理软件 mac找不到启动磁盘 启动磁盘没有足够的空间来进行分区

当你一早打开苹果电脑&#xff0c;结果系统突然提示&#xff1a; “启动磁盘已满&#xff0c;需要删除部分文件”。你会怎么办&#xff1f;如果你认为单纯靠清理废纸篓或者删除大型文件就能释放你的启动磁盘上的空间&#xff0c;那就大错特错了。其实苹果启动磁盘的清理技巧有很…

Nginx基础(06)

Nginx基础&#xff08;05&#xff09; uWSGI 介绍 uWSGI 是一个 Web服务器 主要用途是将Web应用程序部署到生产环境中 可以用来连接Nginx服务与Python动态网站 1. 用 uWSGI 部署 Python 网站项目 配置 Nginx 使其可以将动态访问转交给 uWSGI 安装 python 工具及依赖 安…

Android 日历-周视图

//获取当天 Calendar calendar Calendar.getInstance(); int year calendar.get(Calendar.YEAR); int month calendar.get(Calendar.MONTH) 1; // 注意&#xff1a;月份是从0开始的 int day calendar.get(Calendar.DAY_OF_MONTH); //基姆拉尔森计算公式根据日期判断星期几…

STM32之HAL开发——CubeMX配置串行Flash文件系统

配置流程 在开始配置FATFS前&#xff0c;需要提前配置好RCC的时钟&#xff0c;以及时钟的频率&#xff0c;另外还要配置好Debug选项&#xff08;选择串行&#xff09; 选项介绍 文件系统适用于SD卡&#xff0c;Disk磁盘等&#xff0c;需要我们将对应的驱动打开才可以使用。 …

【vue】Pinia-2 安装Pinia,使用store

1. 安装Pinia 在项目路径下执行npm install pinia 在package.json中查看 2. 使用store 在main.js中添加 import { createPinia } from pinia const pinia createPinia()修改createApp方法 最后示例如下&#xff08;三处修改&#xff09; import { createApp } from vue //…

SimpleImputer缺失数据处理报错解决方案

作者Toby&#xff0c;来源公众号&#xff1a;Python风控建模&#xff0c;SimpleImputer缺失数据处理报错解决方案 今天有学员反馈缺失值代码报错&#xff0c;由于sklearn缺失值处理的包升级&#xff0c;下面把官网最新的缺失值处理代码奉上。 参考https://scikit-learn.org/st…

Java 查找算法

顺序查找 顺序查找算法是一种最简单的查找算法&#xff0c;它的基本思想是从数组的第一个元素开始&#xff0c;逐个比较&#xff0c;直到找到目标元素或遍历完整个数组。 以下是Java实现的顺序查找算法示例代码&#xff1a; public class SequentialSearch {public static i…

设计模式|访问者模式(Visitor Pattern)

文章目录 结构举例优缺点优点缺点代码示例常见面试题访问者模式是一种行为设计模式,它允许在不改变已有类的情况下定义一组新的操作。 这些操作通常分散在不同的类中,但是希望能够对这些类的对象进行统一的处理。 访问者模式的核心思想是将操作从对象结构中分离出来,使得可以…

请把「睡一个好觉」,当成一天里最重要的事来管理

我发现许多人都有这么一种情况&#xff1a;明明知道睡眠很重要&#xff0c;但却总是有意无意地熬夜。 比如&#xff1a; 给自己排了太多的学习和工作量&#xff0c;一不小心就到了凌晨一两点&#xff1b; 总觉得过去的一天什么都没干&#xff0c;宁愿在网上闲逛&#xff0c;也不…

ChatGPT:提升写作效率的必备工具

ChatGPT无限次数:点击直达 html ChatGPT&#xff1a;提升写作效率的必备工具 作为一名有着10年经验的CSDN网站原创文章优质创作者&#xff0c;写作效率对我来说至关重要。在这篇文章中&#xff0c;我将分享如何利用ChatGPT这一强大工具提升写作效率&#xff0c;以及具体的实…

Nacos源码分析,Nacos如何注册一个服务实例?

作为SpringCloudAlibaba微服务架构实战派上下册和RocketMQ消息中间件实战派上下册的作者&#xff0c;我来给大家带来Nacos源码系列的技术文章。 HTTP方式注册服务实例 Naocs使用InstanceController类的HTTP方法register()提供注册服务实例的功能。 /*** 注册服务实例到注册中…

低代码开发平台权威推荐:创新开发、领跑市场!

Gartner是低代码领域的一家权威机构&#xff0c;该机构常常通过"魔力象限"的研究方法&#xff0c;评选全球范围内IT细分领域的产品&#xff0c;来帮助决策者提供重要的咨询建议。本文盘点了Gartner机构推荐的6款低代码平台&#xff1a;Zoho Creator、Mendix、Oracle、…

Java 中建造者模式,请用代码具体举例

建造者模式是一种创建型设计模式&#xff0c;它允许你创建一个复杂对象的不同部分并将它们组装在一起&#xff0c;以产生最终的对象。以下是一个简单的 Java 示例&#xff0c;演示了建造者模式的用法&#xff1a; // 产品类 class Computer {private String cpu;private String…

JVM结构化体系

目录 目录 1.JVM 简介 1.1. 如何理解 JVM 呢&#xff1f; 1.2. 市场主流 JVM 分析&#xff1f; 1.3. 为什么要学习 JVM&#xff1f; 1.4. 字节码底层是如何执行呢&#xff1f; 如何理解 JIT 呢&#xff1f; 为什么 JVM 中解释执行与编译执行的并存&#xff08;混合模式&…

【C++】类和对象③(类的默认成员函数:拷贝构造函数 | 赋值运算符重载)

&#x1f525;个人主页&#xff1a;Forcible Bug Maker &#x1f525;专栏&#xff1a;C 目录 前言 拷贝构造函数 概念 拷贝构造函数的特性及用法 赋值运算符重载 运算符重载 赋值运算符重载 结语 前言 本篇主要内容&#xff1a;类的6个默认成员函数中的拷贝构造函数…

el-drawer二次封装进行可拖拽

1.想要的效果 鼠标放到上面出现箭头显示可拖拽得图标 2.代码实现 2.1封装成自定义指令 // drawerDragDirective.js // 定义指令 const drawerDragDirective {// 指令绑定时的处理函数bind(el, ) {const minWidth 300;const dragDom el.querySelector(.el-drawer);// 创…