亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么在C++中實現函數重載和默認參數

發布時間:2021-05-16 15:19:12 來源:億速云 閱讀:237 作者:Leah 欄目:編程語言

本篇文章給大家分享的是有關怎么在C++中實現函數重載和默認參數,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。


函數名相同,參數個數不同、參數類型不同、參數順序不同

例如下面就是函數重載

void sum(int a, int b){
 cout << a+b << endl;
}

void sum(int a, double b){
 cout << a+b << endl;
}

返回值類型與函數重載無關

返回值類型與函數重載無關,下面代碼不構成重載,編譯會報錯

//返回值類型與函數重載無關
int func(){
 return 0;
}

double func(){
 return 0;
}

實參的隱式類型轉換可能會產生二義性

不同編譯器有不同處理
下面代碼在vs上編譯不過,但是在Xcode中可以編譯通過。

#include "iostream"
using namespace std;

void sum(double a){
 cout << a << endl;
}

void sum(int a){
 cout << a << endl;
}

int main(){
 sum(10);
 
 return 0;
}

函數重載的本質

采用了name mangling或者叫name decoration技術

  • C++編譯器默認會對符號名(比如函數名)進行改編、修飾,有些地方翻譯為“命名傾軋”

  • 重載時會生成多個不同的函數名,不同編譯器(MSVC、g++)有不同的生成規則

  • 通過IDA打開【VS_Release_禁止優化】可以看到 或者通過hopper查看

源碼

下面的代碼

#include "iostream"
using namespace std;

void sum(double a){
 cout << a << endl;
}

void sum(int a){
 cout << a << endl;
}

int main(){
 return 0;
}

在代碼中, void sum(double a){} 和 void sum(int a){} 是如何重載,調用函數的時候是如何能正確找到對應的函數呢?

匯編

我是用xcode的編譯出可執行文件,放在hopper中查看

     __Z3sumd:        // sum(double)
0000000100000ce0         push       rbp         ; CODE XREF=_main+23
0000000100000ce1         mov        rbp, rsp
0000000100000ce4         sub        rsp, 0x10
0000000100000ce8         mov        rdi, qword [__ZNSt3__14coutE_100001000]
0000000100000cef         movsd      qword [rbp+var_8], xmm0
0000000100000cf4         movsd      xmm0, qword [rbp+var_8]
0000000100000cf9         call       imp___stubs___ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEd ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(double)
0000000100000cfe         mov        rdi, rax
0000000100000d01         lea        rsi, qword [__ZNSt3__1L4endlIcNS_11char_traitsIcEEEERNS_13basic_ostreamIT_T0_EES7_]
0000000100000d08         call       __ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEPFRS3_S4_E ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(std::__1::basic_ostream<char, std::__1::char_traits<char> >& (*)(std::__1::basic_ostream<char, std::__1::char_traits<char> >&))
0000000100000d0d         mov        qword [rbp+var_10], rax
0000000100000d11         add        rsp, 0x10
0000000100000d15         pop        rbp
0000000100000d16         ret
                        ; endp
0000000100000d17         nop        word [rax+rax]

可知 void sum(double a){} 被編譯器修改為函數 __Z3sumd

                     __Z3sumi:        // sum(int)
0000000100000da0         push       rbp
0000000100000da1         mov        rbp, rsp
0000000100000da4         sub        rsp, 0x10
0000000100000da8         mov        rax, qword [__ZNSt3__14coutE_100001000]
0000000100000daf         mov        dword [rbp+var_4], edi
0000000100000db2         mov        esi, dword [rbp+var_4]
0000000100000db5         mov        rdi, rax
0000000100000db8         call       imp___stubs___ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEi ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(int)
0000000100000dbd         mov        rdi, rax                                    ; argument #1 for method __ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEPFRS3_S4_E
0000000100000dc0         lea        rsi, qword [__ZNSt3__1L4endlIcNS_11char_traitsIcEEEERNS_13basic_ostreamIT_T0_EES7_]
0000000100000dc7         call       __ZNSt3__113basic_ostreamIcNS_11char_traitsIcEEElsEPFRS3_S4_E ; std::__1::basic_ostream<char, std::__1::char_traits<char> >::operator<<(std::__1::basic_ostream<char, std::__1::char_traits<char> >& (*)(std::__1::basic_ostream<char, std::__1::char_traits<char> >&))
0000000100000dcc         mov        qword [rbp+var_10], rax
0000000100000dd0         add        rsp, 0x10
0000000100000dd4         pop        rbp
0000000100000dd5         ret
                        ; endp
0000000100000dd6         nop        word [cs:rax+rax]

可知 void sum(int a){} 被編譯器修改為函數 __Z3sumi

這樣當我們調用的時候

int main(){
 sum(10.5);
 return 0;
}

匯編如下,可知:因為 10.5是double類型,調用函數的時候是調用 __Z3sumd

0000000100000de0         push       rbp
0000000100000de1         mov        rbp, rsp
0000000100000de4         sub        rsp, 0x10
0000000100000de8         movsd      xmm0, qword [0x100000f80]
0000000100000df0         mov        dword [rbp+var_4], 0x0
0000000100000df7         call       __Z3sumd        ; sum(double)
0000000100000dfc         xor        eax, eax
0000000100000dfe         add        rsp, 0x10
0000000100000e02         pop        rbp
0000000100000e03         ret
                        ; endp
0000000100000e04         nop        word [cs:rax+rax]
0000000100000e0e         nop

函數重載結論

由上面的匯編代碼可知,當參數類型不同的時候,編譯器會生成不同的函數名作為區別,這樣就能實現函數重載。

默認參數

規則

C++允許函數設置默認參數,在調用時可以根據情況省略實參。規則如下:

  • 默認參數只能按照右到左的順序

  • 如果函數同時有聲明、實現,默認參數只能放在函數聲明中

  • 默認參數的值可以是常量、全局符號(全局變量、函數名)

用法:如果函數的實參經常是同一個值,可以考慮使用默認參數

#include "iostream"
using namespace std;
void test(){
 cout << "test()" << endl;
}
// test2函數
// a沒有默認值
// b 默認值是 10
// 最后一個參數默認值是個函數
void test2(int a, int b = 10, void (*func)() = test){
 cout << "a is " << a << endl;
 cout << "b is " << b << endl;
 func();
}
int main(){
 test2(3);
 return 0;
}

可能有沖突,二義性

函數重載、默認參數可能會產生沖突、二義性(建議優先選擇使用默認參數)
例如下面的代碼中, 調用 test(3); 會報錯,因為不知道要執行哪個函數。

#include "iostream"
using namespace std;

void test(int a){
 cout << a << endl;
}

void test(int a,int b = 10){
 cout << a << endl;
}

int main(){
 test(3); // 這里報錯,因為不知道要執行哪個函數
 test(10,20); //這一句可以正確
 return 0;
}

以上就是怎么在C++中實現函數重載和默認參數,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

c++
AI

江山市| 罗城| 郧西县| 青海省| 宁晋县| 乌鲁木齐县| 临清市| 桃园市| 乐清市| 沧州市| 太保市| 玛沁县| 冷水江市| 鹰潭市| 崇明县| 潼关县| 旌德县| 财经| 灌云县| 海伦市| 雅安市| 南汇区| 罗田县| 勐海县| 柏乡县| 南宫市| 五大连池市| 翁源县| 色达县| 荣昌县| 吴堡县| 图木舒克市| 同德县| 凤凰县| 北安市| 阳高县| 西宁市| 邵阳县| 和顺县| 屯昌县| 罗源县|