前言
我們在平時寫代碼中會用到幾個函數但是他們的實作功能相同,但是有些細節卻不同。例如:交換兩個數的值其中包括(int, float,char,double)這些個類型。在C語言中我們是利用不同的函數名來加以區分。
void Swap1(int* a, int* b);
void Swap2(float* a, float* b);
void Swap3(char* a, char* b);
void Swap4(double* a, double* b);
我們可以看出這樣的代碼不美觀而且給程式猿也帶來了很多的不便。于是在C++中人們提出了用一個函數名定義多個函數,也就是所謂的函數重載。
函數重載指的是一個作用域内的幾個函數名字相同但是形參清單不同。這些函數執行操作類似,但是接受的形參類型不一樣,編譯器會根據傳遞的實參類型選擇對應的函數調用。本文将簡單介紹C++中的函數重載。
定義重載函數
假設有一個計算圖形面積的函數,它可以是計算三角形,圓形或正方形的面積。函數的名字都相同,隻是根據傳入的圖形類型來選擇不同的函數來計算面積,程式清單如下:
#include
using namespace std;
typedef struct Triangle//定義三角形結構
{
double high;//高
double baseLen;//底邊長
}Triangle;
typedef struct Circle //定義圓形結構
{
double radius;//半徑
}Circle;
typedef struct Square//定義正方形結構
{
double sideLen;//邊長
}Square;
//函數1.計算三角形面積
double calcArea(const Triangle&)
{
cout<
}
//函數2.計算圓形面積
double calcArea(const Circle&)
{
cout<
return 0;
}
//函數3,計算三角形面積
double calcArea(const Square&)
{
cout<
}
int main(void)
{
Triangle triangle;
Circle circle;
Square square;
calcArea(triangle);//調用函數1
calcArea(circle);//調用函數2
calcArea(square);//調用函數3
return 0;
}
可以看到,定義的三個函數名calcArea都相同,隻是形參類型不同。當分别傳入三角形,圓形和正方形類型時,會調用對應的函數。
運作結果如下:
calcute triangle area
calcute circle area
calcute square area
可以看到,當分别傳入Triangle ,Circle,Square類型時,分别調用了對應的函數。
為什麼要重載
函數重載在一定程式上可以減輕程式員起名字的負擔。最常見的一個例子就是構造函數的重載。
class Test
{
public:
Test(void); // 無參構造函數
Test(int a);//構造函數
Test(int a,int b);//兩個整型參數的構造函數
};
可以看到,類Test的三個構造函數名都為Test。如果沒有重載,要實作三個構造函數就可能需要三個不同的構造函數名區分,這也就增加了類的使用者的負擔,使用者需要傳入不同參數構造對象時,就需要使用不同的構造函數名稱。而有函數重載之後,類的使用者可以使用同一個函數名傳入不同的參數即可。
當然了,如果單純地為了減輕起名字的負擔而去使用函數重載,而使得函數失去了本來的資訊,則是一個不明智的選擇。我們可以為那些操作确實極其相似的函數進行重載。
不能重載的情況
以下幾種情況下,是不能重載或者說是非法的。
main函數不能重載
這是在C++ 11标準中說明的:
A program shall contain a global function called main, which is the designated start of the program....
This function shall not be overloaded.
試想如果作為使用者程式入口函數的main函數被重載了,那麼加載的時候該以哪個為入口呢?
隻有傳回值不同
例如下面兩個聲明隻有傳回值不同,函數名和形參都相同:
double calcArea(const Square&);
int calcArea(const Square&); //非法,僅有傳回值不同,不可重載
試想一下,當你傳入Square類型參數,而不去使用傳回值時,應該調用上面的哪個函數呢?
形參清單看似不同,實則相同
例如使用typedef給Triangle起了一個“别名”:
typedef Triangle MyTri;
double calcArea(const Triangle&);
double calcArea(const MyTri&);
上面這種情況的形參看似不一樣,本質上來說它們并沒有什麼不同。
形參名不同
例如:
double calcArea(const Circle &circle );//形參名為circle
double calcArea(const Circle& cir);//形參名為cir
double calcArea(const Circle& );//省略形參名
這裡形參的名字僅僅是起到說明或者記憶的作用,是以對于上面三個聲明,它們的形參名可以随意起,但不會影響形參清單的内容。
僅有頂層const的差異
例如:
double calcArea(const Circle);//函數1
double calcArea(Circle);//重複聲明了函數1
double calcArea(Circle* const);//函數2
double calcArea(Circle*);//重複聲明了函數2
但需要特别注意的是,如果形參是指針或引用,是可以通過區分指向大到底是常量對象還是非常量對象來實作函數重載。例如下面的情況是可以實作函數重載的:
double calcArea(const Circle&);//作用于常量引用
double calcArea(Circle&);//
double calcArea(const Circle*);//作用于常量指針
double calcArea(Circle*);
總結
在定義了重載函數後,我們需要以合理的實參進行調用。大多數情況下,我們很容易判斷傳入的對應實參需要調用哪個函數,但是有些時候卻并不那麼容易。我們将會在後面的文章中看到如何進行函數比對。
我們對前面的内容做一個總結:
函數重載能夠減輕程式員命名的負擔,但這不應該以丢失可讀性為代價。
main函數不能重載。
重載函數的形參在數量或者類型上要有不同。
不能以傳回值作為函數重載要素。
C語言沒有函數重載。
總結
以上就是這篇文章的全部内容了,希望本文的内容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對我們的支援。