- 函数重载
- 你得先知道它是啥吧?
- 为了明白底层菜鸡大学生甚至打开了Linux
- 最后
幸福往往是学的简单,摸得透彻。
函数重载 你得先知道它是啥吧?菜鸡大学生点了一根烟,说起了从前。
由于菜鸡大学生不是科班出身,但是还要求写c程序设计的实验作业。
里面要实现一个类似计算器的东西,但是由于c不太智能,导致菜鸡大学生写了好几个函数才兼顾了所有的情况,还给它们起了很多不同的名字,十分麻烦,后来学了c++,发现里面的函数重载可以完美解决当时的问题。(今天又是嫌弃c的一天呢)
什么是函数重载呢,在同一个作用域里面,可以写几个同名的函数执行类似的功能,这些同名函数的 形参列表 ( 参数个数 或 类型 或 (有不同类型的参数的)顺序 ) 必须不同。
我们用经典的add函数来举个例子:
- 参数个数不同:
int Add(int a, int b)
{
return a + b;
}
int Add(int a, int b, int c)
{
return a + b + c;
}
- 类型不同:
int add(int a,int b)
{
return a + b;
}
double add(double a,double b)
{
return a + b;
}
- (有不同类型的参数的)顺序不同:
double add(double a,int b)
{
return a + b;
}
double add(int a,double b)
{
return a + b;
}
先传的是int还是double,这个是有区别的。
为了明白底层菜鸡大学生甚至打开了Linux注意:返回值不同是不可以的,原因我们下一个部分再讲。
本部分知识需要一些[[预处理]]相关知识,快去复习。
我就不去细讲辣,我很多朋友都写过相关博客,都可以看。
我们现在linux里面建6个文件,这里我gitee里面的截图:
代码如下:
//add.h
#include
using namespace std;
int add(int a,int b);
double add(double a,double b);
void print(int a,double b);
//add.cpp
#include "add.h"
int add(int a,int b)
{
return a+b;
}
double add(double a,double b)
{
return a+b;
}
//test.cpp
#include "add.h"
int main()
{
int a=1,b=2;
double c=1.1,d=2.2;
cout<<add(a,b)<<endl;
cout<<add(c,d)<<endl;
return 0;
}
c的代码与cpp的相似,只不过去掉了cpp语法改成了c的语法,但是没有去掉重载。
这里就不放出了。
有兴趣的可以去我gitee康一康:函数重载演示
然后编译:
可以很清楚的看出c是不支持函数重载的。
既然知道了cpp行,我们也需要知道cpp为什么行,我们可以去康康汇编,
- 执行指令:
objdump -S CppTest
- 找到重载的函数:
可以发现,这两个add函数的名字不一样。
我就直接用结论去推吧。
- _Z+数字表明函数名字长度,比如add就是_Z3.
- add是原本的函数名。
- 后面的是参数类型,ii就是(int,int),dd就是(double,double)。
我们用这个结论再推一个:
void print(int a,double b)
{
cout<<a<<" "<<b<<" "<<endl;
}
是不是应该是_Z5printid
?
整挺好。
那c呢?
我们去掉重载函数,康康只有一个add函数汇编的名字是什么。
就叫add。
如果能重载,编译器怎么知道你想要哪个add呢?
最后,我们看一下cpp的main函数的调用情况:
你看,c++会知道应该调用哪个函数,因为函数名保存了函数的参数相关的数据。
菜鸡大学生的笔记:由于汇编代码里面没有函数返回值的信息,所有只修改函数返回值是不构成函数重载的。
接着,链接会汇总各文件生成的符号表,就可以找到这些函数对应的地址了。(不记得的去看看预处理知识哦)
最后拖更的原因都是Linux,你相信我。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)