当前位置:首页 » 《随便一记》 » 正文

【初识C++】(缺省参数和函数重载)

19 人参与  2023年04月29日 15:54  分类 : 《随便一记》  评论

点击全文阅读


文章目录

一、缺省参数1.缺省参数定义2.缺省参数分类2.1全缺省参数2.2半缺省参数 二、函数重载1.函数重载概念2.构成重载的几种方式为什么会有函数重载及其原理


一、缺省参数

1.缺省参数定义

缺省参数是在函数的声明中给定参数一个指定的值。
如果传参没有给定参数,那就按照声明中默认的缺省值,如果给定了参数,那就按照给定的参数值。
比如:

using namespace std;void Func(int a = 0){cout << a << endl;}int main(){Func(); // 没有传参时,使用参数的默认值Func(10); // 传参时,使用指定的实参return 0;}

对于Func函数如果没有给定参数,那就按照默认的缺省值来赋值。如果给定了参数,那就按照给定的参数进行赋值。

2.缺省参数分类

2.1全缺省参数

全缺省参数是所有的参数都给定缺省值。

void Func(int a = 10, int b = 20, int c = 30){  cout<<"a = "<<a<<endl;  cout<<"b = "<<b<<endl;  cout<<"c = "<<c<<endl;}

比如上面的函数参数就是全缺省参数。

2.2半缺省参数

半缺省参数是给定部分缺省值,不一定是给定一半。

void Func(int a, int b = 10, int c = 20){  cout<<"a = "<<a<<endl;  cout<<"b = "<<b<<endl;  cout<<"c = "<<c<<endl;}

注意:1.给定半缺省参数必须从右往左给!!!
注意:1.给定半缺省参数必须从右往左给!!!
注意:1.给定半缺省参数必须从右往左给!!!

比如:

void Func(int a =10, int b = 10, int c){  cout<<"a = "<<a<<endl;  cout<<"b = "<<b<<endl;  cout<<"c = "<<c<<endl;}

这样是错误的

也不能间隔着给,比如:

void Func(int a = 10 , int b, int c = 20){  cout<<"a = "<<a<<endl;  cout<<"b = "<<b<<endl;  cout<<"c = "<<c<<endl;}

这样也是错误的。

注意:2.也不能在声明和定义中同时给定缺省值!!
注意:2.也不能在声明和定义中同时给定缺省值!!
注意:2.也不能在声明和定义中同时给定缺省值!!

二、函数重载

1.函数重载概念

函数重载是在同一作用域中具有几个功能相同且同名的函数,这些函数中具有不同的参数(包括参数类型,参数数量,参数位置)。

2.构成重载的几种方式

2.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;}

这两个函数构成重载,因为两个函数同名,且它们的参数类型不同,上面的Add函数参数都是int类型,下面的Add函数参数是double类型。

2.2 参数个数不同构成重载

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

这两个函数构成重载,因为函数名相同,并且第一个fun函数参数个数为0,第二个fun函数参数个数为int a,为1个。

2.3函数参数类型顺序不同

// 3、参数类型顺序不同void fun(int a, char b){cout << "fun(int a,char b)" << endl;}void fun(char b, int a){cout << "fun(char b, int a)" << endl;}

这两个函数构成重载,因为函数名相同,并且第一个fun函数的参数类型分别为 int ,char,但是第二个fun函数的参数类型分别为char,int。

注意:下面的两个函数不构成重载!

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

变量名与函数是否重载无关!

注意:
1.必须是相同的函数名
2.返回值不作为函数重载的条件。也就是返回值与函数是否重载无关。

比如:

int fun(int a){cout << "fun(int a)" << endl;return a+b;}void fun(int a){cout << "fun(int a)" << endl;}int main(){fun(1)fun(2)}

假如要调用函数时,编译器就不知道该找哪个了。

为什么会有函数重载及其原理

对于C语言来说,是不支持函数重载的,C语言不允许出现两个同名的函数。

为了弥补这一缺陷,C++设计成可以使用同名函数来解决C语言的缺陷。

函数重载的原理:

我们知道.c文件生成.exe文件会分成几步:
1、 .c文件经过预处理器进行预处理生成 .i文件。

2、.i文件再经过编译器进行编译后,生成.s文件,该阶段会进行一些语法分析,词法分析,语义分析,符号汇总等操作。

3、.s文件再经过汇编器进行汇编操作后生成.o文件,该阶段会将代码翻译成汇编指令,再翻译成机器码。
在汇编阶段,会生成一张符号表,这张符号存储着被初始化了的全局变量,未被初始化的全局变量,变量名,变量的字节偏移量,函数名,还有地址等。

4、链接阶段,.o文件经过链接器的链接后,会生成.exe可执行程序。在这个阶段,链接器会将不同的.o文件的符号表等数据进行合并和重定位等。

**所以:函数名的地址是在汇编完成后生成的。**所以对于不同的函数名。比如:
在这里插入图片描述

在main.c文件对sum这个函数进行了声明,但是声明不会获得该函数的地址,而在sum.c文件中,经过了链接器的连接之后,将sum.o和main.o文件进行了符号表的合并,获取到了sum这个函数的地址,从而对该函数进行调用。

C++编译器中,对于同名的函数,在编译阶段,C++的编译器会对这两个同名的函数进行函数名修饰

不同的编译器对函数名的修饰规则不同,以Linux的gcc(c语言编译器)和g++(c++编译器)两个编译器为例。

gcc编译器对图中的函数的编译结果为右边:

在这里插入图片描述
g++编译器对图中的函数的编译结果为右边:
在这里插入图片描述

可以明显地看到,g++对函数做了修饰,编译后不再是原来的函数名了。

g++修饰规则:【_Z+函数长度+函数名+类型首字母】

在linux下,采用g++编译完成后,函数名字的修饰发生改变,编译器将函数参
数类型信息添加到修改后的名字中。

通过这里就理解了C语言没办法支持重载,因为同名函数没办法区分。而C++是通过函数修饰规则来区分,只要参数不同,修饰出来的名字就不一样,就支持了重载。


点击全文阅读


本文链接:http://zhangshiyu.com/post/60135.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

关于我们 | 我要投稿 | 免责申明

Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1