網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
寫在前面
關(guān)于C語(yǔ)言的編譯與鏈接不懂的可以看一下下面的文章,先回顧一下以前的知識(shí)。
1 函數(shù)重載的概念
函數(shù)重載:是函數(shù)的一種特殊情況,C++允許在同一作用域中聲明幾個(gè)功能類似的同名函數(shù),這些同名函數(shù)的形參列表(參數(shù)個(gè)數(shù) 或 類型 或 順序)必須不同,常用來(lái)處理實(shí)現(xiàn)功能類似數(shù)據(jù)類型不同的問(wèn)題。
//1.函數(shù)的參數(shù)個(gè)數(shù)不同 #include <iostream> using namespace std; void print() { cout << "print()" << endl; } void print(int a) { cout << "print(int a)" << endl; } int main() { print(); print(1); return 0; }
執(zhí)行結(jié)果如下:
print()
print(int a)
請(qǐng)按任意鍵繼續(xù). . .
//2.函數(shù)的參數(shù)類型不同 #include <iostream> using namespace std; void print(double a) { cout << "print(double a)" << endl; } void print(int a) { cout << "print(int a)" << endl; } int main() { print(1.1); print(1); return 0; }
執(zhí)行結(jié)果如下:
print(double a)
print(int a)
請(qǐng)按任意鍵繼續(xù). . .
//3.函數(shù)的參數(shù)順序不同 #include <iostream> using namespace std; void print(double a, int b) { cout << "print(double a, int b)" << endl; } void print(int b, double a) { cout << "print(int b, double a)" << endl; } int main() { print(1.1, 1); print(1, 1.1); return 0; }
執(zhí)行結(jié)果如下:
print(double a, int b)
print(int b, double a)
請(qǐng)按任意鍵繼續(xù). . .
上面就是支持函數(shù)重載的三種情況,緊接著看如下兩個(gè)函數(shù)是否構(gòu)成函數(shù)重載?
int Add(int num1, int num2) { return num1 + num2; } double Add(int num1,int num2) { return num1 + num2; } int main() { return 0; }
我們一編譯,編譯器就會(huì)報(bào)如下錯(cuò)誤:
通過(guò)上面的分析,我們可以發(fā)現(xiàn)是否構(gòu)成函數(shù)重載只與這些同名函數(shù)的形參列表(參數(shù)個(gè)數(shù) 或 類型 或 順序)有關(guān),與函數(shù)的返回值的類型無(wú)關(guān)。因此返回值不同,不能構(gòu)成函數(shù)重載,在調(diào)用時(shí)無(wú)法區(qū)分。
下面思考如下兩個(gè)函數(shù)是否構(gòu)成函數(shù)重載?
void print(int a) { cout << "print()" << endl; } void print(int a = 0) { cout << "print(int a = 0)" << endl; }
我們一編譯,編譯器就會(huì)報(bào)如下錯(cuò)誤:
因此函數(shù)參數(shù)缺省值不同,也不構(gòu)成函數(shù)重載。
最后再看如下兩個(gè)函數(shù)是否構(gòu)成函數(shù)重載?
void print(int a) { cout << "print()" << endl; } void print(int a = 0) { cout << "print(int a = 0)" << endl; }
很顯然上面兩個(gè)函數(shù)是構(gòu)成函數(shù)重載的,我們編譯也沒(méi)有任何問(wèn)題,但是我們不傳參調(diào)用就會(huì)出問(wèn)題,比如print(),就會(huì)在調(diào)用時(shí)出現(xiàn)歧義。
2 函數(shù)重載原理
通過(guò)上面的學(xué)習(xí),我們現(xiàn)在對(duì)函數(shù)重載的語(yǔ)法有了一定的認(rèn)識(shí)和理解,緊接著我們帶著如下問(wèn)題來(lái)分析一下函數(shù)重載的原理。
為什么C++支持函數(shù)重載,而C語(yǔ)言不支持函數(shù)重載呢?
首先我們?cè)贚inux底下創(chuàng)建三個(gè)文件,來(lái)驗(yàn)證上面的問(wèn)題,如下:
//func.h #include <stdio.h> void f(); void f(int a); //func.c void f() { printf("f()\n"); } void f(int a) { printf("f(int a)\n"); } //test.c #include "func.h" int main() { f(); f(1); return 0; }
調(diào)用C的編譯器編譯test.c和func.c就會(huì)報(bào)如下錯(cuò)誤:
因此,驗(yàn)證了C語(yǔ)言不支持函數(shù)重載。因?yàn)榫幾g的時(shí)候 ,兩個(gè)重載函數(shù),函數(shù)名相同,在func.o的符號(hào)表中存在歧義和沖突。其次,鏈接的時(shí)候也存在歧義和沖突,因?yàn)樗鼈兌际怯煤瘮?shù)名去標(biāo)識(shí)和查找,而重載函數(shù),函數(shù)名相同。
為了驗(yàn)證上面的說(shuō)法,我們屏蔽一個(gè)函數(shù),調(diào)用C的編譯器編譯test.c和func.c,在linux底下會(huì)生成一個(gè)a.out的可執(zhí)行程序,用objdump -S 來(lái)查看一下這個(gè)文件:
同理,我們把剛屏蔽的函數(shù)取消掉,由于C++是兼容C的,因此上面的程序我們可以用C++的編譯器去編譯,其結(jié)果如下
用objdump -S 來(lái)查看一下a.out:
這里不難看出c++目標(biāo)文件中的符號(hào)表中不是直接用函數(shù)名來(lái)標(biāo)識(shí)和查找函數(shù)。而是引入了函數(shù)名修飾規(guī)則,不同編譯器下的函數(shù)名修飾規(guī)則不同。
g++的函數(shù)名修飾規(guī)則:_Z + 函數(shù)名長(zhǎng)度 + 函數(shù)名 + 參數(shù)類型首字母。有了函數(shù)名修飾規(guī)則,只要函數(shù)參數(shù)不同,在func.o符號(hào)表里面重載的函數(shù)就不存在二義性和沖突了。
其次,鏈接的時(shí)候,test.o里面的main函數(shù)去調(diào)用兩個(gè)重載函數(shù)時(shí),去符號(hào)表里面查找地址時(shí)也是明確的。
總結(jié)
原文鏈接:https://blog.csdn.net/m0_50655389/article/details/122901262
相關(guān)推薦
- 2022-12-14 PostgreSQL模式匹配與正則表達(dá)式方法總結(jié)_PostgreSQL
- 2022-05-03 C#設(shè)計(jì)模式之工廠模式_C#教程
- 2022-10-13 react-router?v6實(shí)現(xiàn)動(dòng)態(tài)路由實(shí)例_React
- 2022-07-15 Android?Flutter繪制扇形圖詳解_Android
- 2023-03-18 Python?lambda匿名函數(shù)深入講解_python
- 2022-06-29 Python?高級(jí)教程之線程進(jìn)程和協(xié)程的代碼解析_python
- 2022-10-22 Go語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之二叉樹必會(huì)知識(shí)點(diǎn)總結(jié)_Golang
- 2022-08-21 golang類型推斷與隱式類型轉(zhuǎn)換_Golang
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過(guò)濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支