在實際開發中,有時候我們需要實現幾個功能類似的函數,只是有些細節不同。例如希望交換兩個變量的值,這兩個變量有多種類型,可以是 int、float、char、bool 等,我們需要通過參數把變量的地址傳入函數內部。在C語言中,程序員往往需要分別設計出三個不同名的函數,其函數原型與下面類似:
void swap1(int *a, int *b); //交換 int 變量的值
void swap2(float *a, float *b); //交換 float 變量的值
void swap3(char *a, char *b); //交換 char 變量的值
void swap4(bool *a, bool *b); //交換 bool 變量的值
但在C++中,這完全沒有必要。C++ 允許多個函數擁有相同的名字,只要它們的參數列表不同就可以,這就是函數的重載(Function Overloading)。借助重載,一個函數名可以有多種用途。
參數列表又叫參數簽名,包括參數的類型、參數的個數和參數的順序,只要有一個不同就叫做參數列表不同。
【示例】借助函數重載交換不同類型的變量的值:
#include <iostream>
using namespace std;
//交換 int 變量的值
void Swap(int *a, int *b){
int temp = *a;
*a = *b;
*b = temp;
}
//交換 float 變量的值
void Swap(float *a, float *b){
float temp = *a;
*a = *b;
*b = temp;
}
//交換 char 變量的值
void Swap(char *a, char *b){
char temp = *a;
*a = *b;
*b = temp;
}
//交換 bool 變量的值
void Swap(bool *a, bool *b){
char temp = *a;
*a = *b;
*b = temp;
}
int main(){
//交換 int 變量的值
int n1 = 100, n2 = 200;
Swap(&n1, &n2);
cout<<n1<<", "<<n2<<endl;
//交換 float 變量的值
float f1 = 12.5, f2 = 56.93;
Swap(&f1, &f2);
cout<<f1<<", "<<f2<<endl;
//交換 char 變量的值
char c1 = 'A', c2 = 'B';
Swap(&c1, &c2);
cout<<c1<<", "<<c2<<endl;
//交換 bool 變量的值
bool b1 = false, b2 = true;
Swap(&b1, &b2);
cout<<b1<<", "<<b2<<endl;
return 0;
}
運行結果:
200, 100
56.93, 12.5
B, A
1, 0
本例之所以使用
Swap
這個函數名,而不是使用
swap
,是因為 C++ 標准庫已經提供了交換兩個變量的值的函數,它的名字就是
swap
,位於
algorithm
頭文件中,為了避免和標准庫中的
swap
沖突,本例特地將
S
大寫。
既然標准庫已經提供了 swap() 函數,本例為何又要自己實現一遍呢,這不是費力不討好嗎?交換兩個變量的值是一個經典且實用的函數重載案例,本例這樣做僅僅是為了教學演示,並不是要替代標准庫中的 swap(),讀者在以後的編碼過程中也應該堅持使用標准庫中的 swap()。
通過本例可以發現,重載就是在一個作用范圍內(同一個類、同一個命名空間等)有多個名稱相同但參數不同的函數。重載的結果是讓一個函數名擁有了多種用途,使得命名更加方便(在中大型項目中,給變量、函數、類起名字是一件讓人苦惱的問題),調用更加靈活。
在使用重載函數時,同名函數的功能應當相同或相近,不要用同一函數名去實現完全不相干的功能,雖然程序也能運行,但可讀性不好,使人覺得莫名其妙。
注意,參數列表不同包括參數的個數不同、類型不同或順序不同,僅僅參數名稱不同是不可以的。函數返回值也不能作為重載的依據。
函數的重載的規則:
-
函數名稱必須相同。
-
參數列表必須不同(個數不同、類型不同、參數排列順序不同等)。
-
函數的返回類型可以相同也可以不相同。
-
僅僅返回類型不同不足以成為函數的重載。
C++ 是如何做到函數重載的
C++代碼在編譯時會根據參數列表對函數進行重命名,例如
void Swap(int a, int b)
會被重命名為
_Swap_int_int
,
void Swap(float x, float y)
會被重命名為
_Swap_float_float
。當發生函數調用時,編譯器會根據傳入的實參去逐個匹配,以選擇對應的函數,如果匹配失敗,編譯器就會報錯,這叫做重載決議(Overload Resolution)。
不同的編譯器有不同的重命名方式,這裡僅僅舉例說明,實際情況可能並非如此。
從這個角度講,函數重載僅僅是語法層面的,本質上它們還是不同的函數,占用不同的內存,入口地址也不一樣。