C++中函数重载分析
文章目錄
- 1 重載的概念
- 2 C++中的函數(shù)重載
- 2.1 函數(shù)重載的基本概念
- 2.2 函數(shù)重載的條件
- 2.3 函數(shù)默認遇上函數(shù)重載
- 2.4 編譯器調(diào)用重載函數(shù)的準則
- 3 重載與指針
- 3.1 函數(shù)重載與函數(shù)指針
- 4 類中的函數(shù)重載
- 4.1 類中的重載
- 5 重載的深度意義
- 5.1 重載的意義
1 重載的概念
重載(Overload): 同一個標識符在不同的上下文有不同的意義。
如:
- “洗”和不同的詞匯搭配有不同的含義:洗衣服、洗腦、洗臉、洗車、洗馬桶……
- “play”和不同的單詞搭配有不同的含義:play chess, play pinao, play basketball……
思考:重載在自然語言中是隨處可見的,那么程序設(shè)計中是否也有重載呢?
函數(shù)重載。
2 C++中的函數(shù)重載
2.1 函數(shù)重載的基本概念
函數(shù)重載(Function Overload):
- 用同一個函數(shù)定義不同的函數(shù)。
- 當函數(shù)名和不同的參數(shù)搭配時函數(shù)的含義不同。
2.2 函數(shù)重載的條件
函數(shù)重載至少滿足下面的一個條件 :
- 參數(shù)個數(shù)不同;
- 參數(shù)類型不同;
- 參數(shù)順序不同。
上面的兩個函數(shù)可以構(gòu)成重載函數(shù)。
2.3 函數(shù)默認遇上函數(shù)重載
當函數(shù)默認參數(shù)遇上函數(shù)重載會發(fā)生什么?
函數(shù)默認參數(shù)VS函數(shù)重載:
2.4 編譯器調(diào)用重載函數(shù)的準則
規(guī)則如下:
- 將所有同名函數(shù)作為候選者;
- 嘗試尋找可行的候選參數(shù):
- 精確匹配實參;
- 通過默認參數(shù)能夠匹配實參;
- 通過默認類型轉(zhuǎn)換匹配實參。
匹配失敗:
- 最終尋找到的候選函數(shù)不唯一,則出現(xiàn)二義性,編譯失敗。
- 無法匹配所有候選者,函數(shù)未定義,編譯失敗。
說明:如果能夠精確匹配實參(包含默認參數(shù)),那么就不會再去嘗試通過類型轉(zhuǎn)換匹配實參。這里是需要注意的地方!
函數(shù)重載的注意事項:
- 重載函數(shù)在本質(zhì)上是相互獨立的不同函數(shù);
- 重載函數(shù)的函數(shù)類型不同;
- 函數(shù)返回值不能作為函數(shù)重載的依據(jù);
- 函數(shù)重載是由函數(shù)名和參數(shù)列表決定的。
- 函數(shù)重載必然發(fā)生在同一個作用域中。
編程實驗:函數(shù)重載的本質(zhì)
#include <stdio.h>int add(int a, int b) // int(int, int) {return a + b; }int add(int a, int b, int c) // int(int, int, int) {return a + b + c; }int main() {printf("%p\n", (int(*)(int, int))add);printf("%p\n", (int(*)(int, int, int))add);return 0; } // 查看編譯中間文件符號表的工具:vs命令行:DUMPBIN /symbols pathname(xx.obj)3 重載與指針
3.1 函數(shù)重載與函數(shù)指針
下面的函數(shù)指針將保存哪個函數(shù)的地址?
注意:
void fun(int a, int b = 123);
void (*p)(int a) = fun;
//如上直接編譯錯誤(VS中)
函數(shù)重載遇上函數(shù)指針:
將重載函數(shù)名賦值給函數(shù)指針時:
- 根據(jù)重載規(guī)則挑選與函數(shù)指針參數(shù)列表一致的候選者。
- 嚴格匹配候選者的函數(shù)類型與函數(shù)指針的函數(shù)類型。
注意:不會跟默認參數(shù)扯上關(guān)系。
還需要注意以下幾點:
- 函數(shù)重載必然發(fā)生在同一個作用域中;
- 編譯器需要用參數(shù)列表或函數(shù)類型進行函數(shù)選擇;
- 無法直接通過函數(shù)名得到重載函數(shù)的入口地址。
4 類中的函數(shù)重載
4.1 類中的重載
類中的成員函數(shù)可以進行重載:
- 構(gòu)造函數(shù)的重載。
- 普通成員函數(shù)的重載。
- 靜態(tài)成員函數(shù)的重載。
問題:全局函數(shù)、普通成員函數(shù)以及靜態(tài)成員函數(shù)之間是否可以構(gòu)成重載?
為了回答這個問題,我們有必要把函數(shù)重載的本質(zhì)再拿出來:
所以上面的答案是:由于不在同一個作用域,所以全局函數(shù)和類中的普通成員函數(shù)、靜態(tài)成員函數(shù)之間不構(gòu)成重載。而普通的成員函數(shù)和靜態(tài)成員函數(shù)之間可以構(gòu)成重載。
示例代碼:
#include <stdio.h>class Test {int i; public:Test(){printf("Test::Test()\n");this->i = 0;}Test(int i){printf("Test::Test(int i)\n");this->i = i;}Test(const Test& obj){printf("Test(const Test& obj)\n");this->i = obj.i;}static void func(){printf("void Test::func()\n");}void func(int i){printf("void Test::func(int i), i = %d\n", i);}int getI(){return i;} };void func() {printf("void func()\n"); }void func(int i) {printf("void func(int i), i = %d\n", i); }int main() {func();func(1);Test t; // Test::Test()Test t1(1); // Test::Test(int i)Test t2(t1); // Test(const Test& obj)func(); // void func()Test::func(); // void Test::func()func(2); // void func(int i), i = 2;t1.func(2); // void Test::func(int i), i = 2t1.func(); // void Test::func()return 0; }5 重載的深度意義
5.1 重載的意義
意義如下:
- 通過函數(shù)名對函數(shù)功能進行提示。
- 通過參數(shù)列表對函數(shù)用法進行提示。
- 擴展系統(tǒng)中已經(jīng)存在的函數(shù)功能。
重載的意義分析:
#include <stdio.h> #include <string.h>char* strcpy(char* buf, const char* str, unsigned int n) {return strncpy(buf, str, n); }int main() {const char* s = "D.T.Software";char buf[8] = {0};//strcpy(buf, s);strcpy(buf, s, sizeof(buf)-1);printf("%s\n", buf);return 0; }重載能夠擴展系統(tǒng)中已經(jīng)存在的函數(shù)功能,重載也能夠擴展更多的功能,操作符重載就是最重要的功能之一!
參考資料:
總結(jié)
以上是生活随笔為你收集整理的C++中函数重载分析的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。