亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關(guān)于我們
? 蟲蟲下載站

?? 第8章 c++函數(shù)的高級特性.txt

?? C C++ JAVA等軟件方面的面試題目
?? TXT
字號:
第8章 C++函數(shù)的高級特性
對比于C語言的函數(shù),C++增加了重載(overloaded)、內(nèi)聯(lián)(inline)、const和virtual四種新機制。其中重載和內(nèi)聯(lián)機制既可用于全局函數(shù)也可用于類的成員函數(shù),const與virtual機制僅用于類的成員函數(shù)。 
重載和內(nèi)聯(lián)肯定有其好處才會被C++語言采納,但是不可以當(dāng)成免費的午餐而濫用。本章將探究重載和內(nèi)聯(lián)的優(yōu)點與局限性,說明什么情況下應(yīng)該采用、不該采用以及要警惕錯用。

8.1 函數(shù)重載的概念
8.1.1 重載的起源

自然語言中,一個詞可以有許多不同的含義,即該詞被重載了。人們可以通過上下文來判斷該詞到底是哪種含義。“詞的重載”可以使語言更加簡練。例如“吃飯”的含義十分廣泛,人們沒有必要每次非得說清楚具體吃什么不可。別迂腐得象孔已己,說茴香豆的茴字有四種寫法。

在C++程序中,可以將語義、功能相似的幾個函數(shù)用同一個名字表示,即函數(shù)重載。這樣便于記憶,提高了函數(shù)的易用性,這是C++語言采用重載機制的一個理由。例如示例8-1-1中的函數(shù)EatBeef,EatFish,EatChicken可以用同一個函數(shù)名Eat表示,用不同類型的參數(shù)加以區(qū)別。



void EatBeef(…); // 可以改為 void Eat(Beef …);

void EatFish(…); // 可以改為 void Eat(Fish …);

void EatChicken(…); // 可以改為 void Eat(Chicken …);




示例8-1-1 重載函數(shù)Eat


C++語言采用重載機制的另一個理由是:類的構(gòu)造函數(shù)需要重載機制。因為C++規(guī)定構(gòu)造函數(shù)與類同名(請參見第9章),構(gòu)造函數(shù)只能有一個名字。如果想用幾種不同的方法創(chuàng)建對象該怎么辦?別無選擇,只能用重載機制來實現(xiàn)。所以類可以有多個同名的構(gòu)造函數(shù)。


8.1.2 重載是如何實現(xiàn)的?

幾個同名的重載函數(shù)仍然是不同的函數(shù),它們是如何區(qū)分的呢?我們自然想到函數(shù)接口的兩個要素:參數(shù)與返回值。

如果同名函數(shù)的參數(shù)不同(包括類型、順序不同),那么容易區(qū)別出它們是不同的函數(shù)。

如果同名函數(shù)僅僅是返回值類型不同,有時可以區(qū)分,有時卻不能。例如:

void Function(void);

int Function (void);

上述兩個函數(shù),第一個沒有返回值,第二個的返回值是int類型。如果這樣調(diào)用函數(shù):

int x = Function ();

則可以判斷出Function是第二個函數(shù)。問題是在C++/C程序中,我們可以忽略函數(shù)的返回值。在這種情況下,編譯器和程序員都不知道哪個Function函數(shù)被調(diào)用。

所以只能靠參數(shù)而不能靠返回值類型的不同來區(qū)分重載函數(shù)。編譯器根據(jù)參數(shù)為每個重載函數(shù)產(chǎn)生不同的內(nèi)部標(biāo)識符。例如編譯器為示例8-1-1中的三個Eat函數(shù)產(chǎn)生象_eat_beef、_eat_fish、_eat_chicken之類的內(nèi)部標(biāo)識符(不同的編譯器可能產(chǎn)生不同風(fēng)格的內(nèi)部標(biāo)識符)。


如果C++程序要調(diào)用已經(jīng)被編譯后的C函數(shù),該怎么辦?

假設(shè)某個C函數(shù)的聲明如下:

void foo(int x, int y);

該函數(shù)被C編譯器編譯后在庫中的名字為_foo,而C++編譯器則會產(chǎn)生像_foo_int_int之類的名字用來支持函數(shù)重載和類型安全連接。由于編譯后的名字不同,C++程序不能直接調(diào)用C函數(shù)。C++提供了一個C連接交換指定符號extern“C”來解決這個問題。例如:

extern “C”

{

void foo(int x, int y);

… // 其它函數(shù)

}

或者寫成

extern “C”

{

#include “myheader.h”

… // 其它C頭文件

}

這就告訴C++編譯譯器,函數(shù)foo是個C連接,應(yīng)該到庫中找名字_foo而不是找_foo_int_int。C++編譯器開發(fā)商已經(jīng)對C標(biāo)準(zhǔn)庫的頭文件作了extern“C”處理,所以我們可以用#include 直接引用這些頭文件。


注意并不是兩個函數(shù)的名字相同就能構(gòu)成重載。全局函數(shù)和類的成員函數(shù)同名不算重載,因為函數(shù)的作用域不同。例如:

void Print(…); // 全局函數(shù)

class A

{…

void Print(…); // 成員函數(shù)

}

不論兩個Print函數(shù)的參數(shù)是否不同,如果類的某個成員函數(shù)要調(diào)用全局函數(shù)Print,為了與成員函數(shù)Print區(qū)別,全局函數(shù)被調(diào)用時應(yīng)加‘::’標(biāo)志。如

::Print(…); // 表示Print是全局函數(shù)而非成員函數(shù)

8.1.3 當(dāng)心隱式類型轉(zhuǎn)換導(dǎo)致重載函數(shù)產(chǎn)生二義性

示例8-1-3中,第一個output函數(shù)的參數(shù)是int類型,第二個output函數(shù)的參數(shù)是float類型。由于數(shù)字本身沒有類型,將數(shù)字當(dāng)作參數(shù)時將自動進(jìn)行類型轉(zhuǎn)換(稱為隱式類型轉(zhuǎn)換)。語句output(0.5)將產(chǎn)生編譯錯誤,因為編譯器不知道該將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 << " 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! ambiguous call, 因為自動類型轉(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++程序員必須要搞清楚概念,否則錯誤將防不勝防。


8.2.1 重載與覆蓋

成員函數(shù)被重載的特征:

(1)相同的范圍(在同一個類中);

(2)函數(shù)名字相同;

(3)參數(shù)不同;

(4)virtual關(guān)鍵字可有可無。

覆蓋是指派生類函數(shù)覆蓋基類函數(shù),特征是:

(1)不同的范圍(分別位于派生類與基類);

(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 Base

{

public:

void f(int x){ cout << "Base::f(int) " << x << endl; }

void f(float x){ cout << "Base::f(float) " << x << endl; }

virtual void g(void){ cout << "Base::g(void)" << endl;}

};



class Derived : public Base

{

public:

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); // 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ù)不同。此時,不論有無virtual關(guān)鍵字,基類的函數(shù)將被隱藏(注意別與重載混淆)。

(2)如果派生類的函數(shù)與基類的函數(shù)同名,并且參數(shù)也相同,但是基類函數(shù)沒有virtual關(guān)鍵字。此時,基類的函數(shù)被隱藏(注意別與覆蓋混淆)。

示例程序8-2-2(a)中:

(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 Base

{

public:

virtual void f(float x){ cout << "Base::f(float) " << x << endl; }

void g(float x){ cout << "Base::g(float) " << x << endl; }

void h(float x){ cout << "Base::h(float) " << x << endl; }

};

class Derived : public Base

{

public:

virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }

void g(int x){ cout << "Derived::g(int) " << x << endl; }

void h(float x){ cout << "Derived::h(float) " << x << endl; }

};


示例8-2-2(a)成員函數(shù)的重載、覆蓋和隱藏


據(jù)作者考察,很多C++程序員沒有意識到有“隱藏”這回事。由于認(rèn)識不夠深刻,“隱藏”的發(fā)生可謂神出鬼沒,常常產(chǎn)生令人迷惑的結(jié)果。

示例8-2-2(b)中,bp和dp指向同一地址,按理說運行結(jié)果應(yīng)該是相同的,可事實并非這樣。


void main(void)

{

Derived d;

Base *pb = &d;

Derived *pd = &d;

// Good : behavior depends solely on type of the object

pb->f(3.14f); // Derived::f(float) 3.14 

pd->f(3.14f); // Derived::f(float) 3.14


// Bad : behavior depends on type of the pointer

pb->g(3.14f); // Base::g(float) 3.14 

pd->g(3.14f); // Derived::g(int) 3 (surprise!)


// Bad : behavior depends on type of the pointer

pb->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)化為字符串,所以在編譯時出錯。


class Base

{

public:

void f(int x);

};

class Derived : public Base

{

public:

void f(char *str);

};

void Test(void)

{

Derived *pd = new Derived;

pd->f(10); // error

}


示例8-2-3 由于隱藏而導(dǎo)致錯誤


從示例8-2-3看來,隱藏規(guī)則似乎很愚蠢。但是隱藏規(guī)則至少有兩個存在的理由:

u 寫語句pd->f(10)的人可能真的想調(diào)用Derived::f(char *)函數(shù),只是他誤將參數(shù)寫錯了。有了隱藏規(guī)則,編譯器就可以明確指出錯誤,這未必不是好事。否則,編譯器會靜悄悄地將錯就錯,程序員將很難發(fā)現(xiàn)這個錯誤,流下禍根。

u 假如類Derived有多個基類(多重繼承),有時搞不清楚哪些基類定義了函數(shù)f。如果沒有隱藏規(guī)則,那么pd->f(10)可能會調(diào)用一個出乎意料的基類函數(shù)f。盡管隱藏規(guī)則看起來不怎么有道理,但它的確能消滅這些意外。


示例8-2-3中,如果語句pd->f(10)一定要調(diào)用函數(shù)Base::f(int),那么將類Derived修改為如下即可。

class Derived : public Base

{

public:

void f(char *str);

void f(int x) { Base::f(x); }

};

8.3 參數(shù)的缺省值
有一些參數(shù)的值在每次函數(shù)調(diào)用時都相同,書寫這樣的語句會使人厭煩。C++語言采用參數(shù)的缺省值使書寫變得簡潔(在編譯時,缺省值由編譯器自動插入)。

參數(shù)缺省值的使用規(guī)則:

l 【規(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) // 錯誤,缺省值出現(xiàn)在函數(shù)的定義體中

{


}

為什么會這樣?我想是有兩個原因:一是函數(shù)的實現(xiàn)(定義)本來就與參數(shù)是否有缺省值無關(guān),所以沒有必要讓缺省值出現(xiàn)在函數(shù)的定義體中。二是參數(shù)的缺省值可能會改動,顯然修改函數(shù)的聲明比修改函數(shù)的定義要方便。


l 【規(guī)則8-3-2】如果函數(shù)有多個參數(shù),參數(shù)只能從后向前挨個兒缺省,否則將導(dǎo)致函數(shù)調(diào)用語句怪模怪樣。

正確的示例如下:

void Foo(int x, int y=0, int z=0);

錯誤的示例如下:

void Foo(int x=0, int y, int z=0); 


要注意,使用參數(shù)的缺省值并沒有賦予函數(shù)新的功能,僅僅是使書寫變得簡潔一些。它可能會提高函數(shù)的易用性,但是也可能會降低函數(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;

// output(x); // error! ambiguous call

output(x,y); // output int 1 and float 0.5

}




示例8-3-2 參數(shù)的缺省值將導(dǎo)致重載函數(shù)產(chǎn)生二義性

8.4 運算符重載
8.4.1 概念

在C++語言中,可以用關(guān)鍵字operator加上運算符來表示函數(shù),叫做運算符重載。例如兩個復(fù)數(shù)相加函數(shù):

Complex Add(const Complex &a, const Complex &b);

可以用運算符重載來表示:

Complex operator +(const Complex &a, const Complex &b);

運算符與普通函數(shù)在調(diào)用時的不同之處是:對于普通函數(shù),參數(shù)出現(xiàn)在圓括號內(nèi);而對于運算符,參數(shù)出現(xiàn)在其左、右側(cè)。例如

Complex a, b, c;


c = Add(a, b); // 用普通函數(shù)

c = a + b; // 用運算符 +

如果運算符被重載為全局函數(shù),那么只有一個參數(shù)的運算符叫做一元運算符,有兩個參數(shù)的運算符叫做二元運算符。

如果運算符被重載為類的成員函數(shù),那么一元運算符沒有參數(shù),二元運算符只有一個右側(cè)參數(shù),因為對象自己成了左側(cè)參數(shù)。

從語法上講,運算符既可以定義為全局函數(shù),也可以定義為成員函數(shù)。文獻(xiàn)[Murray , p44-p47]對此問題作了較多的闡述,并總結(jié)了表8-4-1的規(guī)則。


運算符
規(guī)則

所有的一元運算符
建議重載為成員函數(shù)

= () [] ->
只能重載為成員函數(shù)

+= -= /= *= &= |= ~= %= >>= <<=
建議重載為成員函數(shù)

所有其它運算符
建議重載為全局函數(shù)


表8-4-1 運算符的重載規(guī)則


?? 快捷鍵說明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
欧美亚洲国产一卡| av激情综合网| 欧美高清精品3d| 亚洲成人激情自拍| 欧美一区二区三区啪啪| 蜜臀国产一区二区三区在线播放| 日韩欧美你懂的| 国产专区综合网| 中文一区二区在线观看| 91美女片黄在线| 亚洲成av人影院在线观看网| 欧美精品 日韩| 狠狠v欧美v日韩v亚洲ⅴ| 久久精品在线免费观看| 波多野洁衣一区| 亚洲午夜三级在线| 日韩一级二级三级| 丰满少妇久久久久久久| 1024国产精品| 91精品国产日韩91久久久久久| 黑人巨大精品欧美一区| 国产精品二区一区二区aⅴ污介绍| 色婷婷一区二区| 日本欧美一区二区在线观看| 久久精品一区二区三区av| 91小视频免费看| 丝袜国产日韩另类美女| 国产午夜亚洲精品理论片色戒 | 性做久久久久久免费观看| 日韩精品一区在线观看| 99这里只有精品| 麻豆成人91精品二区三区| 国产精品视频九色porn| 欧美裸体一区二区三区| 成人高清免费观看| 免费一区二区视频| 亚洲色图欧美偷拍| 精品卡一卡二卡三卡四在线| 色综合天天天天做夜夜夜夜做| 免费久久99精品国产| 中文字幕一区日韩精品欧美| 日韩欧美www| 91免费国产视频网站| 久久成人综合网| 免费欧美在线视频| 一区二区中文视频| 精品88久久久久88久久久| 日本韩国欧美一区| 国产高清一区日本| 理论电影国产精品| 亚洲成人一区在线| 亚洲日本免费电影| 久久精品人人做人人爽人人 | 欧美日韩一级视频| 成人精品在线视频观看| 麻豆91在线播放| 调教+趴+乳夹+国产+精品| 亚洲少妇30p| 国产视频在线观看一区二区三区 | 久久国产剧场电影| 日日夜夜精品视频天天综合网| 中文字幕一区二区三区四区 | 国产午夜精品一区二区| 欧美电影一区二区| 欧美视频在线一区| 在线观看国产日韩| 色婷婷精品久久二区二区蜜臀av| 波多野结衣91| 成人avav影音| 成人高清av在线| 豆国产96在线|亚洲| 国产99一区视频免费 | 亚洲大尺度视频在线观看| 一区二区三区久久| 一区二区三区影院| 国产精品久久久久久久久免费桃花 | 久久久激情视频| 26uuu精品一区二区在线观看| 欧美一卡二卡三卡| 欧美一级日韩不卡播放免费| 欧美一区二区三区精品| 欧美一级一区二区| 日韩欧美在线123| 精品国产一区二区精华| 精品成人在线观看| 国产欧美视频一区二区三区| 久久久久国色av免费看影院| 国产喷白浆一区二区三区| 国产亚洲一区二区三区| 亚洲国产精品99久久久久久久久 | 亚洲天堂精品在线观看| 亚洲欧美另类小说视频| 亚洲尤物视频在线| 婷婷综合久久一区二区三区| 国产精品综合视频| 成人高清视频在线观看| 色噜噜狠狠一区二区三区果冻| 91国模大尺度私拍在线视频| 欧美日韩一区在线观看| 日韩欧美国产精品| 欧美激情一区在线观看| 亚洲免费看黄网站| 日韩电影在线观看一区| 国产一区二区三区在线观看免费| 成人开心网精品视频| 色妹子一区二区| 日韩精品一区二区三区中文不卡| 久久久久高清精品| 一区二区欧美国产| 蜜桃视频一区二区| 不卡一区二区三区四区| 欧美色窝79yyyycom| 精品国内二区三区| 成人免费在线观看入口| 三级影片在线观看欧美日韩一区二区| 国内外精品视频| 一本色道久久综合亚洲精品按摩| 在线综合视频播放| 国产精品久久久久影院老司| 亚洲超碰97人人做人人爱| 国产一区二区三区四区五区美女| 99久久久久久99| 日韩一级二级三级| 亚洲欧美一区二区三区国产精品| 日本人妖一区二区| 97aⅴ精品视频一二三区| 欧美成人三级电影在线| 亚洲欧美电影院| 国产电影精品久久禁18| 欧美精品一二三区| 亚洲欧美自拍偷拍色图| 精品一区二区综合| 欧美午夜电影一区| 国产精品视频一二三| 久久超碰97人人做人人爱| 色欧美乱欧美15图片| 久久久久久久久蜜桃| 亚洲成人黄色影院| 91在线高清观看| 久久久91精品国产一区二区精品| 性做久久久久久| 色偷偷久久人人79超碰人人澡| 久久精品亚洲精品国产欧美kt∨ | 麻豆精品久久精品色综合| 91看片淫黄大片一级在线观看| 久久综合九色综合97婷婷| 亚洲成人三级小说| 色偷偷成人一区二区三区91 | 亚洲三级理论片| 国产精品性做久久久久久| 欧美一级片在线看| 丝袜亚洲另类欧美| 欧美日韩一区小说| 亚洲精品ww久久久久久p站| 色欧美88888久久久久久影院| 国产欧美日韩视频在线观看| 精品中文av资源站在线观看| 欧美高清激情brazzers| 亚洲成人动漫在线观看| 色呦呦一区二区三区| 中文字幕中文字幕一区二区 | 国产精品日韩精品欧美在线| 国产综合成人久久大片91| 欧美一级xxx| 伦理电影国产精品| 91麻豆精品国产91| 日韩电影在线免费观看| 欧美挠脚心视频网站| 亚洲123区在线观看| 欧美卡1卡2卡| 五月激情六月综合| 在线播放欧美女士性生活| 午夜成人免费电影| 欧美精品在线一区二区三区| 五月婷婷另类国产| 91精品婷婷国产综合久久竹菊| 爽爽淫人综合网网站| 欧美一区二区三区视频在线| 日产国产欧美视频一区精品| 欧美大片在线观看| 国产精品888| 国产无一区二区| 成人免费观看男女羞羞视频| 中文字幕在线不卡国产视频| 色狠狠一区二区| 日韩av在线发布| 日韩精品一区二区三区视频播放 | 午夜伦欧美伦电影理论片| 欧美日韩在线一区二区| 日韩国产欧美在线观看| 精品欧美久久久| 成人免费观看av| 亚洲综合丁香婷婷六月香| 欧美久久一二三四区| 久久精品理论片| 国产精品青草综合久久久久99| 91视频观看视频| 青青青伊人色综合久久| 久久久精品中文字幕麻豆发布| bt7086福利一区国产|