【C++】函数重载及实现原理

这篇具有很好参考价值的文章主要介绍了【C++】函数重载及实现原理。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

       🔥🔥 欢迎来到小林的博客!!
      🛰️博客主页:✈️林 子
      🛰️博客专栏:✈️ 小林C++之路
      🛰️社区 :✈️ 进步学堂
      🛰️欢迎关注:👍点赞🙌收藏✍️留言

什么是函数重载?

函数重载就是允许出现多个同名函数,但是参数类型,参数顺序,参数数量不同才能构成函数重载。

我们先来看看以下这段代码,用C语言编译会怎么样。

#include<stdio.h>
void fun(int i)
{
	printf("fun(int i )\n");
}

void fun(int i, double d)
{
	printf("fun(int i, double d)\n");
}

int main()
{
	fun(1);c
	fun(1, 1.1);
}

我们会发现出现这样的错误。

【C++】函数重载及实现原理

这是因为C语言再检测时发现了同名函数。那如果我们把文件改成cpp形式再编译呢?

我们会发现cpp可以打印出来。

【C++】函数重载及实现原理

这是因为C++支持函数重载,而函数重载就是可以同时存在多个函数名相同的函数,但是必须至少满足以下三点的其中一点。

1. 函数参数数量不同

2. 函数参数类型不同

3. 函数参数顺序不同

例如:

//以下两个fun函数数量,类型不同
void fun(int i)
{
	printf("fun(int i )\n");
}
void fun(int i, double d)
{
	printf("fun(int i, double d)\n");
}

//以下两个a函数,数量相同,但是参数顺序不同..也可以归纳为类型不同
void a(int i, double d)
{
	;
}
void a(double d, int i)
{
	;
}
//以下两个b函数,参数类型不同
void b(int i) {
	;
}
void b(double d)
{
	;
}

为什么要有函数重载?

我们知道了函数重载是什么之后,那么为什么要有函数重载?比如我要写两个函数,一个函数可以对整形进行加法运算,而另一个函数要对浮点型进行加法运算。那么我们在C语言中只能这样写。

#include<stdio.h>
int addI(int x, int y)
{
	return x + y;
}
double addD(double x, double y)
{
	return x + y;
}

int main()
{
	addI(10,20);
	addD(10.5, 20.5);
}

但是在C++中,支持函数重载,我们可以这样写。

#include<stdio.h>
int add(int x, int y)
{
	return x + y;
}
double add(double x, double y)
{
	return x + y;
}

int main()
{
	add(10,20);
	add(10.5, 20.5);
}

有没有发现支持了函数重载之后,add 有自动类型推导 的作用? 这就是函数重载的作用。避免对各种各样功能相同,但是类型不同的函数取不同的函数名。这样记起来也麻烦,用了函数重载之后,那么只需要记住一个函数即可。

函数重载的原理

那么函数重载是如何实现的呢?那我们要知道.c/cpp 源程序在生成可执行程序时要经历的四个阶段。

1.预处理 : 负责头文件展开,宏替换,条件编译,去掉注释等…

2. 编译 : 检查语法,语法没有错误后,把C代码转换为汇编指令

3. 汇编 : 将汇编指令转换为二进制机器码

4.链接 : 合并符号表

那么我们就在每个阶段来观察一下这个函数,接下来将会在linux系统下演示。

首先,我们先写一个fun.c的源文件

#include<stdio.h>

void fun(int i)
{
  printf("fun(int i )\n");
}

void fun(int i ,double d)
{
  printf("fun (int i , double d)\n");
}

int main()
{
  fun(1);
  fun(1,1.1);
  return 0;
}

然后执行命令 gcc -E fun.c -o fun.i 将fun.c进行预处理后生成一个.i文件

然后我们发现编译器并没有报错,那么说明冲突不是在预处理阶段产生的。我们来看看fun.i的内容。

【C++】函数重载及实现原理

我们会发现,和C语言代码并没有其他区别,区别就是头文件展开了。

接下来,我们再执行 gcc -S fun.i -o fun.s 将预处理过的.i文件进行编译,转换为汇编代码。

【C++】函数重载及实现原理

然后我们就会发现编译出错,那么说明冲突是编译时产生的。那么我们换成.cpp文件来看。

我们把fun.c 改成funcpp.cpp, 然后分别执行 g++ -E funcpp.cpp -o funcpp.ig++ -S funcpp.i-o funcpp.s

【C++】函数重载及实现原理

我们会发现编译没有问题,那么我们来查看一下汇编代码。

【C++】函数重载及实现原理

我们发现 fun(int i)函数实际修饰名是_Z3funi。

那么我们再看看fun(int i ,double d)的函数名是什么。

【C++】函数重载及实现原理

fun(int i ,double d)的实际修饰名是 _Z3funid。

在Linux操作系统下, 这个3表示函数名的字符串长度,后面跟上函数名,之后就是参数类型,int 为i,double为d。

也就是说虽然 fun(int i ,double d)fun(int i ,double d)函数名相同,但是在修饰之后。实际上的函数名是_Z3funi_Z3funid。所以并不存在冲突问题。

那么为什么C语言不行呢?那么我们写一个正确的C语言代码,重复上面步骤,看看C语言的实际函数名是什么。

这是一段正确的C语言代码。

#include<stdio.h>

void fun(int i)
{
  printf("fun(int i )\n");
}

int main()
{
  fun(1);
  return 0;
}

随后进行预处理和编译。

编译后我们查看fun.s文件,会发现函数fun实际上就是函数fun。没有其他修饰规则,所以就造成了函数名冲突问题。

【C++】函数重载及实现原理

总结

函数重载本质就是对函数名进行了修饰,允许函数名相同,但是参数数量,顺序,类型不同的函数同时共存。虽然表面上看起来这个函数名是这个函数的函数名,但是在编译后函数名会被处理修饰。所以每个重载的函数真正意义上的函数名是不同的。而C语言则没有加修饰,函数名就是函数名,所以就会出现冲突编译报错的情况。文章来源地址https://www.toymoban.com/news/detail-435620.html

到了这里,关于【C++】函数重载及实现原理的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C++】STL——string的模拟实现、常用构造函数、迭代器、运算符重载、扩容函数、增删查改

    string使用文章   这里我们 实现常用的第四个string(const char* s)和析构函数     拷贝构造函数实现:   在堆上使用new为当前对象的成员变量_str分配内存空间,大小为s._capacity + 1字节,即字符串的容量加上一个结束符\\0的空间。   我们使用深拷贝而不是浅拷贝,

    2024年02月15日
    浏览(55)
  • 【C++】:拷贝构造函数与赋值运算符重载的实例应用之日期类的实现

    🔑日期类的实现,将按以下声明依次进行,其中因为Print函数比较短,直接放到类里面让其变成内联函数 🔑而我们实现日期类经常要用到某月有多少天,在这里先把获得某月有多少天的函数实现出来。实现时先检查传参有没有问题,在注意把数组设置成静态的,出了作用域

    2024年02月08日
    浏览(66)
  • 【C++】中类的6个默认成员函数 取地址及const成员函数 && 学习运算符重载 && 【实现一个日期类】

    1.1 运算符重载【引入】 C++为了增强代码的可读性引入了 运算符重载 ,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。 函数名字为: operator后面接需要重载的运算符符号。 函数原型:

    2024年02月21日
    浏览(49)
  • 【C++】STL——list的模拟实现、构造函数、迭代器类的实现、运算符重载、增删查改

    list使用文章 析构函数   在定义了一个类模板list时。我们让该类模板包含了一个内部结构体_list_node,用于表示链表的节点。该结构体包含了指向前一个节点和后一个节点的指针以及节点的值。在list中保存了链表的头节点指针和链表长度大小。       因为list的底层实现

    2024年02月14日
    浏览(50)
  • 【C++学习】C++入门 | 缺省参数 | 函数重载 | 探究C++为什么能够支持函数重载

    上一篇文章我介绍了C++该怎么学,什么是命名空间,以及C++的输入输出, 这里是传送门:http://t.csdn.cn/Oi6V8 这篇文章我们继续来学习C++的基础知识。 目录 写在前面: 1. 缺省参数 2. 函数重载 3. C++是如何支持函数重载的 写在最后: 在学习C语言的时候,如果一个函数存在参数

    2024年02月13日
    浏览(48)
  • C++ 学习 ::【基础篇:05】:C++ 函数重载认识及使用、简单介绍:C++ 支持函数重载的原因

    本系列 C++ 相关文章 仅为笔者学习笔记记录,用自己的理解记录学习!C++ 学习系列将分为三个阶段: 基础篇、STL 篇、高阶数据结构与算法篇 ,相关重点内容如下: 基础篇 : 类与对象 (涉及C++的三大特性等); STL 篇 : 学习使用 C++ 提供的 STL 相关库 ; 高阶数据结构与算

    2024年02月06日
    浏览(54)
  • 【C++】函数重载 & 引用 & 内联函数

    目录 一,函数重载 1,函数重载的概念 2,C++支持函数重载的原理 二,引用 1,引用概念 2,引用特性 3,常引用 4,做参数 5,做返回值 6,传值、传引用效率比较 7,值和引用的作为返回值类型的性能比较 8,引用和指针的区别 三,内联函数 1,概念 2,特性 自然语言中,一个

    2024年02月08日
    浏览(41)
  • c++ ——命名空间、缺省函数、函数重载

    在C/C++中,变量、函数和后面要学到的类都是大量存在的,这些变量、函数和类的名称将都存 在于全局作用域中,可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化, 以避免命名冲突或名字污染,namespace的出现就是针对这种问题的 简单一句话命

    2024年02月16日
    浏览(52)
  • C++ 面向对象(3)——重载运算符和重载函数

    C++ 允许在同一作用域中的某个 函数 和 运算符 指定多个定义,分别称为 函数重载 和 运算符重载 。 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列表和定义(实现)不相同。 当您调用一个 重载函数 或 重载运算符 时

    2024年02月10日
    浏览(57)
  • C++入门:函数缺省参数与函数重载

    目录 1.函数缺省参数 1.1 缺省参数概念 1.2 缺省参数分类 2.函数重载 2.1 函数重载概念 2.2 C++支持函数重载的原理 缺省参数是 声明或定义函数时 为函数的 参数指定一个缺省值 。在调用该函数时,如果没有指定实 参则采用该形参的缺省值,否则使用指定的实参,有点备胎的意

    2024年02月12日
    浏览(33)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包