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

2024-04-17 00:04

本文主要是介绍【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;
}

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

请添加图片描述

这篇关于【C++对于C语言的扩充】函数重载、引用以及内联函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/910275

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

Python中isinstance()函数原理解释及详细用法示例

《Python中isinstance()函数原理解释及详细用法示例》isinstance()是Python内置的一个非常有用的函数,用于检查一个对象是否属于指定的类型或类型元组中的某一个类型,它是Py... 目录python中isinstance()函数原理解释及详细用法指南一、isinstance()函数

python中的高阶函数示例详解

《python中的高阶函数示例详解》在Python中,高阶函数是指接受函数作为参数或返回函数作为结果的函数,下面:本文主要介绍python中高阶函数的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录1.定义2.map函数3.filter函数4.reduce函数5.sorted函数6.自定义高阶函数

Python内存管理机制之垃圾回收与引用计数操作全过程

《Python内存管理机制之垃圾回收与引用计数操作全过程》SQLAlchemy是Python中最流行的ORM(对象关系映射)框架之一,它提供了高效且灵活的数据库操作方式,本文将介绍如何使用SQLAlc... 目录安装核心概念连接数据库定义数据模型创建数据库表基本CRUD操作创建数据读取数据更新数据删除数据查

Go语言中json操作的实现

《Go语言中json操作的实现》本文主要介绍了Go语言中的json操作的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、jsOChina编程N 与 Go 类型对应关系️ 二、基本操作:编码与解码 三、结构体标签(Struc

Python中的sort方法、sorted函数与lambda表达式及用法详解

《Python中的sort方法、sorted函数与lambda表达式及用法详解》文章对比了Python中list.sort()与sorted()函数的区别,指出sort()原地排序返回None,sor... 目录1. sort()方法1.1 sort()方法1.2 基本语法和参数A. reverse参数B.

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2