C++入門語法之函數(shù)重載詳解
寫在前面
關(guān)于C語言的編譯與鏈接不懂的可以看一下下面的文章,先回顧一下以前的知識。
1 函數(shù)重載的概念
函數(shù)重載:是函數(shù)的一種特殊情況,C++允許在同一作用域中聲明幾個功能類似的同名函數(shù),這些同名函數(shù)的形參列表(參數(shù)個數(shù) 或 類型 或 順序)必須不同,常用來處理實現(xiàn)功能類似數(shù)據(jù)類型不同的問題。
//1.函數(shù)的參數(shù)個數(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)
請按任意鍵繼續(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)
請按任意鍵繼續(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)
請按任意鍵繼續(xù). . .
上面就是支持函數(shù)重載的三種情況,緊接著看如下兩個函數(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;
}
我們一編譯,編譯器就會報如下錯誤:

通過上面的分析,我們可以發(fā)現(xiàn)是否構(gòu)成函數(shù)重載只與這些同名函數(shù)的形參列表(參數(shù)個數(shù) 或 類型 或 順序)有關(guān),與函數(shù)的返回值的類型無關(guān)。因此返回值不同,不能構(gòu)成函數(shù)重載,在調(diào)用時無法區(qū)分。
下面思考如下兩個函數(shù)是否構(gòu)成函數(shù)重載?
void print(int a)
{
cout << "print()" << endl;
}
void print(int a = 0)
{
cout << "print(int a = 0)" << endl;
}
我們一編譯,編譯器就會報如下錯誤:

因此函數(shù)參數(shù)缺省值不同,也不構(gòu)成函數(shù)重載。
最后再看如下兩個函數(shù)是否構(gòu)成函數(shù)重載?
void print(int a)
{
cout << "print()" << endl;
}
void print(int a = 0)
{
cout << "print(int a = 0)" << endl;
}
很顯然上面兩個函數(shù)是構(gòu)成函數(shù)重載的,我們編譯也沒有任何問題,但是我們不傳參調(diào)用就會出問題,比如print(),就會在調(diào)用時出現(xiàn)歧義。
2 函數(shù)重載原理
通過上面的學習,我們現(xiàn)在對函數(shù)重載的語法有了一定的認識和理解,緊接著我們帶著如下問題來分析一下函數(shù)重載的原理。
為什么C++支持函數(shù)重載,而C語言不支持函數(shù)重載呢?
首先我們在Linux底下創(chuàng)建三個文件,來驗證上面的問題,如下:
//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就會報如下錯誤:

因此,驗證了C語言不支持函數(shù)重載。因為編譯的時候 ,兩個重載函數(shù),函數(shù)名相同,在func.o的符號表中存在歧義和沖突。其次,鏈接的時候也存在歧義和沖突,因為它們都是用函數(shù)名去標識和查找,而重載函數(shù),函數(shù)名相同。
為了驗證上面的說法,我們屏蔽一個函數(shù),調(diào)用C的編譯器編譯test.c和func.c,在linux底下會生成一個a.out的可執(zhí)行程序,用objdump -S 來查看一下這個文件:

同理,我們把剛屏蔽的函數(shù)取消掉,由于C++是兼容C的,因此上面的程序我們可以用C++的編譯器去編譯,其結(jié)果如下

用objdump -S 來查看一下a.out:

這里不難看出c++目標文件中的符號表中不是直接用函數(shù)名來標識和查找函數(shù)。而是引入了函數(shù)名修飾規(guī)則,不同編譯器下的函數(shù)名修飾規(guī)則不同。
g++的函數(shù)名修飾規(guī)則:_Z + 函數(shù)名長度 + 函數(shù)名 + 參數(shù)類型首字母。有了函數(shù)名修飾規(guī)則,只要函數(shù)參數(shù)不同,在func.o符號表里面重載的函數(shù)就不存在二義性和沖突了。
其次,鏈接的時候,test.o里面的main函數(shù)去調(diào)用兩個重載函數(shù)時,去符號表里面查找地址時也是明確的。

總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容
相關(guān)文章
關(guān)于C++友元函數(shù)的實現(xiàn)講解
今天小編就為大家分享一篇關(guān)于關(guān)于C++友元函數(shù)的實現(xiàn)講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-12-12
C語言 結(jié)構(gòu)體(Struct)詳解及示例代碼
本文主要介紹C語言 結(jié)構(gòu)體的知識,學習C語言肯定需要學習結(jié)構(gòu)體,這里詳細說明了結(jié)構(gòu)體并附示例代碼,供大家參考學習,有需要的小伙伴可以參考下2016-08-08
C++使用JsonCpp庫操作json格式數(shù)據(jù)示例
這篇文章主要介紹了C++使用JsonCpp庫操作json格式數(shù)據(jù),結(jié)合實例形式詳細分析了JsonCpp庫的下載及C++使用JsonCpp庫對json格式數(shù)據(jù)序列化相關(guān)操作技巧,需要的朋友可以參考下2017-06-06

