




版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
1、對比于C語言的函數(shù),C+增加了重載(overloaded)、內(nèi)聯(lián)(inline)、const和virtual四種新機(jī)制。其中重載和內(nèi)聯(lián)機(jī)制既可用于全局函數(shù)也可用于類的成員函數(shù),const與virtual機(jī)制僅用于類的成員函數(shù)。 重載和內(nèi)聯(lián)肯定有其好處才會(huì)被C+語言采納,但是不可以當(dāng)成免費(fèi)的午餐而濫用。本章將探究重載和內(nèi)聯(lián)的優(yōu)點(diǎn)與局限性,說明什么情況下應(yīng)該采用、不該采用以及要警惕錯(cuò)用。 8.1 函數(shù)重載的概念8.1.1 重載的起源 自然語言中,一個(gè)詞可以有許多不同的含義,即該詞被重載了。人們可以通過上下文來判斷該詞到底是哪種含義?!霸~的重載”可以使語言更加簡練。例如“吃飯”的含義十分廣泛,人們沒
2、有必要每次非得說清楚具體吃什么不可。別迂腐得象孔已己,說茴香豆的茴字有四種寫法。在C+程序中,可以將語義、功能相似的幾個(gè)函數(shù)用同一個(gè)名字表示,即函數(shù)載。這樣便于記憶,提高了函數(shù)的易用性,這是C+語言采用重載機(jī)制的一個(gè)理由。例如示例8-1-1中的函數(shù)EatBeef,EatFish,EatChicken可以用同一個(gè)函數(shù)名Eat表示,用不同類型的參數(shù)加以區(qū)別。void EatBeef(); / 可以改為 void Eat(Beef ); void EatFish(); / 可以改為 void Eat(Fish ); void EatChicken(); / 可以改為 void Eat(Chicken
3、 );示例8-1-1 重載函數(shù)EatC+語言采用重載機(jī)制的另一個(gè)理由是:類的構(gòu)造函數(shù)需要重載機(jī)制。因?yàn)镃+規(guī)定構(gòu)造函數(shù)與類同名(請參見第9章),構(gòu)造函數(shù)只能有一個(gè)名字。如果想用幾種不同的方法創(chuàng)建對象該怎么辦?別無選擇,只能用重載機(jī)制來實(shí)現(xiàn)。所以類可以有多個(gè)同名的構(gòu)造函數(shù)。8.1.2 重載是如何實(shí)現(xiàn)的? 幾個(gè)同名的重載函數(shù)仍然是不同的函數(shù),它們是如何區(qū)分的呢?我們自然想到函數(shù)接口的兩個(gè)要素:參數(shù)與返回值。如果同名函數(shù)的參數(shù)不同(包括類型、順序不同),那么容易區(qū)別出它們是不同的函數(shù)。如果同名函數(shù)僅僅是返回值類型不同,有時(shí)可以區(qū)分,有時(shí)卻不能。例如:void Function(void);int F
4、unction (void);上述兩個(gè)函數(shù),第一個(gè)沒有返回值,第二個(gè)的返回值是int類型。如果這樣調(diào)用函數(shù): int x = Function ();則可以判斷出Function是第二個(gè)函數(shù)。問題是在C+/C程序中,我們可以忽略函數(shù)的返回值。在這種情況下,編譯器和程序員都不知道哪個(gè)Function函數(shù)被調(diào)用。所以只能靠參數(shù)而不能靠返回值類型的不同來區(qū)分重載函數(shù)。編譯器根據(jù)參數(shù)為每個(gè)重載函數(shù)產(chǎn)生不同的內(nèi)部標(biāo)識(shí)符。例如編譯器為示例8-1-1中的三個(gè)Eat函數(shù)產(chǎn)生象_eat_beef、_eat_fish、_eat_chicken之類的內(nèi)部標(biāo)識(shí)符(不同的編譯器可能產(chǎn)生不同風(fēng)格的內(nèi)部標(biāo)識(shí)符)。如果C+程
5、序要調(diào)用已經(jīng)被編譯后的C函數(shù),該怎么辦?假設(shè)某個(gè)C函數(shù)的聲明如下:void foo(int x, int y);該函數(shù)被C編譯器編譯后在庫中的名字為_foo,而C+編譯器則會(huì)產(chǎn)生像_foo_int_int之類的名字用來支持函數(shù)重載和類型安全連接。由于編譯后的名字不同,C+程序不能直接調(diào)用C函數(shù)。C+提供了一個(gè)C連接交換指定符號(hào)extern“C”來解決這個(gè)問題。例如:extern “C” void foo(int x, int y); / 其它函數(shù) 或者寫成extern “C” #include “myheader.h” / 其它C頭文件這就告訴C+編譯譯器,函數(shù)foo是個(gè)C連接,應(yīng)該到庫中找名
6、字_foo而不是找_foo_int_int。C+編譯器開發(fā)商已經(jīng)對C標(biāo)準(zhǔn)庫的頭文件作了extern“C”處理,所以我們可以用include 直接引用這些頭文件。注意并不是兩個(gè)函數(shù)的名字相同就能構(gòu)成重載。全局函數(shù)和類的成員函數(shù)同名不算重載,因?yàn)楹瘮?shù)的作用域不同。例如: void Print(); / 全局函數(shù) class A void Print(); / 成員函數(shù) 不論兩個(gè)Print函數(shù)的參數(shù)是否不同,如果類的某個(gè)成員函數(shù)要調(diào)用全局函數(shù)Print,為了與成員函數(shù)Print區(qū)別,全局函數(shù)被調(diào)用時(shí)應(yīng)加:標(biāo)志。 如:Print(); / 表示Print是全局函數(shù)而非成員函數(shù)8.1.3 當(dāng)心隱式類型轉(zhuǎn)
7、換導(dǎo)致重載函數(shù)產(chǎn)生二義性 示例8-1-3中,第一個(gè)output函數(shù)的參數(shù)是int類型,第二個(gè)output函數(shù)的參數(shù)是float類型。由于數(shù)字本身沒有類型,將數(shù)字當(dāng)作參數(shù)時(shí)將自動(dòng)進(jìn)行類型轉(zhuǎn)換(稱為隱式類型轉(zhuǎn)換)。語句output(0.5)將產(chǎn)生編譯錯(cuò)誤,因?yàn)榫幾g器不知道該將0.5轉(zhuǎn)換成int還是float類型的參數(shù)。隱式類型轉(zhuǎn)換在很多地方可以簡化程序的書寫,但是也可能留下隱患。# include <iostream.h>void output( int x); / 函數(shù)聲明void output( float x); / 函數(shù)聲明void output( int x) cout &l
8、t;< " output int " << x << endl ;void output( float x) cout << " output float " << x << endl ;void main(void) int x = 1; float y = 1.0; output(x); / output int 1 output(y); / output float 1 output(1); / output int 1/ output(0.5); / error! ambiguou
9、s call, 因?yàn)樽詣?dòng)類型轉(zhuǎn)換 output(int(0.5); / output int 0 output(float(0.5); / output float 0.5示例8-1-3 隱式類型轉(zhuǎn)換導(dǎo)致重載函數(shù)產(chǎn)生二義性8.2 成員函數(shù)的重載、覆蓋與隱藏 成員函數(shù)的重載、覆蓋(override)與隱藏很容易混淆,C+程序員必須要搞清楚概念,否則錯(cuò)誤將防不勝防。8.2.1 重載與覆蓋 成員函數(shù)被重載的特征:(1)相同的范圍(在同一個(gè)類中);(2)函數(shù)名字相同;(3)參數(shù)不同;(4)virtual關(guān)鍵字可有可無。 覆蓋是指派生類函數(shù)覆蓋基類函數(shù),特征是:(1)不同的范圍(分別位于派生類與基類);
10、(2)函數(shù)名字相同;(3)參數(shù)相同;(4)基類函數(shù)必須有virtual關(guān)鍵字。 示例8-2-1中,函數(shù)Base:f(int)與Base:f(float)相互重載,而Base:g(void)被Derived:g(void)覆蓋。#include <iostream.h>class Basepublic: void f(int x) cout << "Base:f(int) " << x << endl; void f(float x) cout << "Base:f(float) " <&l
11、t; x << endl; virtual void g(void) cout << "Base:g(void)" << endl;class Derived : public Basepublic: virtual void g(void) cout << "Derived:g(void)" << endl; void main(void) Derived d; Base *pb = &d; pb->f(42); / Base:f(int) 42 pb->f(3.14f);
12、 / Base:f(float) 3.14 pb->g(); / Derived:g(void)示例8-2-1成員函數(shù)的重載和覆蓋8.2.2 令人迷惑的隱藏規(guī)則 本來僅僅區(qū)別重載與覆蓋并不算困難,但是C+的隱藏規(guī)則使問題復(fù)雜性陡然增加。這里“隱藏”是指派生類的函數(shù)屏蔽了與其同名的基類函數(shù),規(guī)則如下:(1)如果派生類的函數(shù)與基類的函數(shù)同名,但是參數(shù)不同。此時(shí),不論有無virtual關(guān)鍵字,基類的函數(shù)將被隱藏(注意別與重載混淆)。(2)如果派生類的函數(shù)與基類的函數(shù)同名,并且參數(shù)也相同,但是基類函數(shù)沒有virtual關(guān)鍵字。此時(shí),基類的函數(shù)被隱藏(注意別與覆蓋混淆)。 示例程序8-2-2(a)
13、中:(1)函數(shù)Derived:f(float)覆蓋了Base:f(float)。(2)函數(shù)Derived:g(int)隱藏了Base:g(float),而不是重載。(3)函數(shù)Derived:h(float)隱藏了Base:h(float),而不是覆蓋。 #include <iostream.h>class Basepublic: virtual void f(float x) cout << "Base:f(float) " << x << endl; void g(float x) cout << "B
14、ase:g(float) " << x << endl; void h(float x) cout << "Base:h(float) " << x << endl; ;class Derived : public Basepublic: virtual void f(float x) cout << "Derived:f(float) " << x << endl; void g(int x) cout << "Derived
15、:g(int) " << x << endl; void h(float x) cout << "Derived:h(float) " << x << endl; ;示例8-2-2(a)成員函數(shù)的重載、覆蓋和隱藏 據(jù)作者考察,很多C+程序員沒有意識(shí)到有“隱藏”這回事。由于認(rèn)識(shí)不夠深刻,“隱藏”的發(fā)生可謂神出鬼沒,常常產(chǎn)生令人迷惑的結(jié)果。示例8-2-2(b)中,bp和dp指向同一地址,按理說運(yùn)行結(jié)果應(yīng)該是相同的,可事實(shí)并非這樣。void main(void)Derived d;Base *pb = &
16、;d;Derived *pd = &d;/ Good : behavior depends solely on type of the objectpb->f(3.14f); / Derived:f(float) 3.14 pd->f(3.14f); / Derived:f(float) 3.14/ Bad : behavior depends on type of the pointerpb->g(3.14f); / Base:g(float) 3.14pd->g(3.14f); / Derived:g(int) 3 (surprise!)/ Bad : be
17、havior depends on type of the pointerpb->h(3.14f); / Base:h(float) 3.14 (surprise!)pd->h(3.14f); / Derived:h(float) 3.14 示例8-2-2(b) 重載、覆蓋和隱藏的比較8.2.3 擺脫隱藏 隱藏規(guī)則引起了不少麻煩。示例8-2-3程序中,語句pd->f(10)的本意是想調(diào)用函數(shù)Base:f(int),但是Base:f(int)不幸被Derived:f(char *)隱藏了。由于數(shù)字10不能被隱式地轉(zhuǎn)化為字符串,所以在編譯時(shí)出錯(cuò)。class Basepublic:
18、void f(int x);class Derived : public Basepublic:void f(char *str);void Test(void)Derived *pd = new Derived;pd->f(10); / error示例8-2-3 由于隱藏而導(dǎo)致錯(cuò)誤從示例8-2-3看來,隱藏規(guī)則似乎很愚蠢。但是隱藏規(guī)則至少有兩個(gè)存在的理由:寫語句pd->f(10)的人可能真的想調(diào)用Derived:f(char *)函數(shù),只是他誤將參數(shù)寫錯(cuò)了。有了隱藏規(guī)則,編譯器就可以明確指出錯(cuò)誤,這未必不是好事。否則,編譯器會(huì)靜悄悄地將錯(cuò)就錯(cuò),程序員將很難發(fā)現(xiàn)這個(gè)錯(cuò)誤,流下禍根。
19、假如類Derived有多個(gè)基類(多重繼承),有時(shí)搞不清楚哪些基類定義了函數(shù)f。如果沒有隱藏規(guī)則,那么pd->f(10)可能會(huì)調(diào)用一個(gè)出乎意料的基類函數(shù)f。盡管隱藏規(guī)則看起來不怎么有道理,但它的確能消滅這些意外。示例8-2-3中,如果語句pd->f(10)一定要調(diào)用函數(shù)Base:f(int),那么將類Derived修改為如下即可。class Derived : public Basepublic:void f(char *str);void f(int x) Base:f(x); ;8.3 參數(shù)的缺省值 有一些參數(shù)的值在每次函數(shù)調(diào)用時(shí)都相同,書寫這樣的語句會(huì)使人厭煩。C+語言采用參數(shù)
20、的缺省值使書寫變得簡潔(在編譯時(shí),缺省值由編譯器自動(dòng)插入)。參數(shù)缺省值的使用規(guī)則:【規(guī)則8-3-1】參數(shù)缺省值只能出現(xiàn)在函數(shù)的聲明中,而不能出現(xiàn)在定義體中。例如:void Foo(int x=0, int y=0); / 正確,缺省值出現(xiàn)在函數(shù)的聲明中>void Foo(int x=0, int y=0) / 錯(cuò)誤,缺省值出現(xiàn)在函數(shù)的定義體中 為什么會(huì)這樣?我想是有兩個(gè)原因:一是函數(shù)的實(shí)現(xiàn)(定義)本來就與參數(shù)是否有缺省值無關(guān),所以沒有必要讓缺省值出現(xiàn)在函數(shù)的定義體中。二是參數(shù)的缺省值可能會(huì)改動(dòng),顯然修改函數(shù)的聲明比修改函數(shù)的定義要方便?!疽?guī)則8-3-2】如果函數(shù)有多個(gè)參數(shù),參數(shù)只能從后向
21、前挨個(gè)兒缺省,否則將導(dǎo)致函數(shù)調(diào)用語句怪模怪樣。正確的示例如下:void Foo(int x, int y=0, int z=0);錯(cuò)誤的示例如下:void Foo(int x=0, int y, int z=0);要注意,使用參數(shù)的缺省值并沒有賦予函數(shù)新的功能,僅僅是使書寫變得簡潔一些。它可能會(huì)提高函數(shù)的易用性,但是也可能會(huì)降低函數(shù)的可理解性。所以我們只能適當(dāng)?shù)厥褂脜?shù)的缺省值,要防止使用不當(dāng)產(chǎn)生負(fù)面效果。示例8-3-2中,不合理地使用參數(shù)的缺省值將導(dǎo)致重載函數(shù)output產(chǎn)生二義性。#include <iostream.h>void output( int x);void output( int x, float y=0.0);void output( int x) cout << " output int " << x << endl ; void output( int x, float y) cout << " output int " << x << " and float " << y << endl ; void main(void) int x=1; float y=0.5;/ ou
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- GB/T 45590-2025向日葵黑莖病菌檢疫鑒定方法
- GB/T 45519-2025紡織品纖維定量分析顯微鏡智能識(shí)別法
- 材料力學(xué)與智能材料性能控制重點(diǎn)基礎(chǔ)知識(shí)點(diǎn)
- 材料疲勞斷裂機(jī)理實(shí)驗(yàn)驗(yàn)證重點(diǎn)基礎(chǔ)知識(shí)點(diǎn)
- 經(jīng)濟(jì)學(xué)理論與現(xiàn)實(shí)的沖突試題及答案
- 銀行發(fā)生火災(zāi)的應(yīng)急預(yù)案(3篇)
- 船上發(fā)生火災(zāi)應(yīng)急預(yù)案(3篇)
- 火災(zāi)觸電踩踏事故專項(xiàng)應(yīng)急預(yù)案(3篇)
- 鐵路超大火災(zāi)應(yīng)急預(yù)案(3篇)
- 高考數(shù)學(xué)間接法探究及試題及答案
- 醫(yī)療護(hù)理與人文關(guān)懷課件
- 用地理知識(shí)介紹美國
- 2024-2025年高考生物一輪復(fù)習(xí)知識(shí)點(diǎn)講解專題3-2細(xì)胞呼吸含解析
- 《生物制品連續(xù)制造指南》
- 保衛(wèi)管理員三級(jí)練習(xí)題
- 湖北荊州市監(jiān)利市暢惠交通投資有限公司招聘筆試沖刺題2024
- 食品配送行業(yè)安全生產(chǎn)管理制度
- 土力學(xué)知到智慧樹章節(jié)測試課后答案2024年秋青島理工大學(xué)
- 手術(shù)室護(hù)理疑難病例討論
- 國家秘密載體的管理要求
- 硫酸安全使用管理及使用制度(4篇)
評論
0/150
提交評論