青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

洛譯小筑

別來無恙,我的老友…
隨筆 - 45, 文章 - 0, 評論 - 172, 引用 - 0
數據加載中……

[ECPP讀書筆記 條目20] 傳參時要多用“引用常量”,少用傳值

默認情況下,C++為函數傳入和傳出對象是采用傳值方式的(這是由C語言繼承而來的特征)。除非你明確使用其他方法,函數的形式參數總會通過復制實在參數的副本來創建,并且,函數的調用者得到的也是函數返回值的一個副本。這些副本是由對象的拷貝構造函數創建的。這使得“傳值”成為一項代價十分昂貴的操作。請觀察下邊的示例中類的層次結構:

class Person {

public:

  Person();                        // 省略參數表以簡化代碼

  virtual ~Person();               // 條目7解釋了它為什么是虛函數

  ...

 

private:

  std::string name;

  std::string address;

};

 

class Student: public Person {

public:

  Student();                       // 再次省略參數表

  virtual ~Student();

  ...

 

private:

  std::string schoolName;

  std::string schoolAddress;

};

請觀察下面的代碼,這里我們調用一個名為validateStudent的函數,通過為這一函數傳進一個Student類型的參數(傳值方式),它將返回這一學生的身份是否合法:

bool validateStudent(Student s);         // 通過傳值方式接受一個Student對象

 

Student plato;                           // 柏拉圖是蘇格拉底的學生

 

 

bool platoIsOK = validateStudent(plato); // 調用這一函數

在這個函數被調用時將會發生些什么呢?

很顯然地,在這一時刻,通過調用Student的拷貝構造函數,可以將這一函數的s參數初始化為plato的值。同樣顯然的是,svalidateStudent返回的時候將被銷毀。所以這一函數中傳參的開銷就是調用一次Student的拷貝構造函數和一次Student的析構函數。

但是上邊的分析僅僅是冰山一角。一個Student對象包含兩個string對象,所以每當你構造一個Student對象時,你都必須構造兩個string對象。同時,由于Student類是從Person類繼承而來,所以在每次構造Student對象時,你都必須再構造一個Person對象。一個Person對象又包含兩個額外的string對象,所以每次對Person的構造還要進行額外的兩次string的構造。最后的結果是,通過傳值方式傳遞一個Student對象會引入以下幾個操作:調用一次Student的拷貝構造函數,調用一次Person的拷貝構造函數,調用四次string的拷貝構造函數。在Student的這一副本被銷毀時,相應的每次構造函數調用都對應著一次析構函數的調用。因此我們看到:通過傳值方式傳遞一個Student對象總體的開銷究竟有多大?竟達到了六次構造函數和六次析構函數的調用!

下面向你介紹正確的方法,這一方法才會使函數擁有期望的行為。畢竟你期望的是所有對象以可靠的方式進行初始化和銷毀。與此同時,如果可以繞過所有這些構造和析構操作將是件很愜意的事情。這個方法就是:通過引用常量傳遞參數:

bool validateStudent(const Student& s);

這樣做效率會提高很多:由于不會創建新的對象,所以就不會存在構造函數或析構函數的調用。改進的參數表中的const是十分重要的。由于早先版本的validateStudent通過傳值方式接收Student參數,所以調用者了解:無論函數對于傳入的Student對象進行什么樣的操作,都不會對原對象造成任何影響,validateStudent僅僅會對對象的副本進行修改。而改進版本中Student對象是以引用形式傳入的,有必要將其聲明為const的,因為如果不這樣,調用者就需要關心傳入validateStudentStudent對象有可能會被修改。

通過引用傳參也可以避免“截斷問題”。當一個派生類的對象以一個基類對象的形式傳遞(傳值方式)時,基類的拷貝構造函數就會被調用,此時,這一對象的獨有特征——使它區別于基類對象的特征會被“截掉”。剩下的只是一個簡單的基類對象,這并不奇怪,因為它是由基類構造函數創建的。這肯定不是你想要的。請看下邊的示例,假設你正在使用一組類來實現一個圖形窗口系統:

class Window {

public:

  ...

  std::string name() const;        // 返回窗口的名字

  virtual void display() const;    // 繪制窗口和內容

};

 

class WindowWithScrollBars: public Window {

public:

  ...

  virtual void display() const;

};

所有的Window對象都有一個名字,可以通過name函數取得。所有的窗口都可以被顯示出來,可以通過調用display實現。display是虛函數,這一事實告訴我們,簡單基類Window的對象與派生出的WindowWithScrollBars對象的顯示方式是不一樣的。(參見條目3436

現在,假設你期望編寫一個函數來打印出當前窗口的名字然后顯示這一窗口。下面是錯誤的實現方法:

void printNameAndDisplay(Window w) // 錯誤! 參數傳遞的對象將被截斷!

{

  std::cout << w.name();

  w.display();

}

考慮一下當你將一個WindowWithScrollBars對象傳入這個函數時將會發生些什么:

WindowWithScrollBars wwsb;

 

printNameAndDisplay(wwsb);

參數w將被構造為一個Window對象——還記得么?它是通過傳值方式傳入的。這里,使wwsb具體化的獨有信息將被截掉。無論傳入函數的對象的具體類型是什么,在printNameAndDisplay的內部,w將總保有一個Window類的對象的身份(因為它本身就是一個Window的對象)。特別地,在printNameAndDisplay內部對display的調用總是Window::display,而永遠不會是WindowWithScrollBars::display

解決截斷問題的方法是:通過引用常量傳參:

void printNameAndDisplay(const Window& w)

{                                  // 工作正常,參數將不會被截斷。

  std::cout << w.name();

  w.display();

}

現在w的類型就是傳入窗口對象的精確類型。

揭開C++編譯器的面紗,你將會發現引用通常情況下是以指針的形式實現的,所以通過引用傳遞通常意味著實際上是在傳遞一個指針。因此,如果傳遞一個內建數據類型的對象(比如int),傳值會被傳遞引用更為高效。那么,對于內建數據類型,當你在傳值和傳遞常量引用之間徘徊時,傳值方式不失為一個更好的選擇。迭代器和STL中的函數對象也是如此,這是因為它們設計的初衷就是能夠更適于傳值,這是C++的慣例。迭代器和函數對象的設計人員有責任考慮復制時的效率問題和截斷問題。(這也是一個“使用哪種規則,取決于當前使用哪一部份的C++”的例子,參見條目1

內建數據類型體積較小,所以一些人得出這樣的結論:所有體積較小的類型都適合使用傳值,即使它們是用戶自定義的。這是一個不可靠的推理。僅僅通過一個對象體積小并不能判定調用它的拷貝構造函數的代價就很低。許多對象——包括大多數STL容器——其中僅僅包含一個指針和很少量的其它內容,但是復制此類對象的同時,它所指向的所有內容都需要復制。這將付出十分高昂的代價。

即使體積較小的對象的拷貝構造函數不會帶來巨大的開銷,它也會引入性能問題。一些編譯器對內建數據類型和用戶自定義數據類型是分別對待的,即使它們的表示方式完全相同。比如說一些編譯器很樂意將一個單純的double值放入寄存器中,這是語言的常規;但將一個僅包含一個double值的對象放入寄存器時,編譯器就會報錯了。當你遇到這種事情時,你可以使用引用傳遞這類對象,因為編譯器此時一定會將指針(引用的具體實現)放入寄存器中。

對于“小型的用戶自定義數據類型不適用于傳值方式”還有一個理由,那就是:作為用戶自定義類型,它們的大小可能會改變。現在很小的類型在未來的版本中可能會變得很大,這是因為它的內部實現方式可能會改變。即使是你更改了C++語言的具體實現都可能會影響到類型的大小。比如,在我編寫上面的示例的時候,一些對標準庫實現中string的大小竟然達到了另一些的七倍。

總體上講,只有內建數據類型、STL迭代器和函數對象類型適用于傳值方式。對于所有其它的類型,都應該遵循本條款中的建議:盡量使用引用常量傳參,而不是傳值。

時刻牢記

盡量使用引用常量傳參,而不是傳值方式。因為一般情況下傳引用更高效,而且可以避免“截斷問題”。

對于內建數據類型、STL迭代和函數對象類型,這一規則就不適用了,對它們來說通常傳值方式更實用。

posted on 2007-06-01 18:12 ★ROY★ 閱讀(1458) 評論(3)  編輯 收藏 引用 所屬分類: Effective C++

評論

# re: 【翻譯】[Effective C++第三版?中文版][第20條]盡量使用“引用常量”傳參,而不是傳值  回復  更多評論   

好漂亮的程序啊!
2007-06-02 09:16 | 深藍色的音符

# re: 【翻譯】[Effective C++第三版?中文版][第20條]盡量使用“引用常量”傳參,而不是傳值  回復  更多評論   

給你做了個鏈接,希望以后能跟你多多交流.
因為我現在也在開始學習C++,不過好難啊!
2007-06-02 13:04 | 深藍色的音符

# re: 【翻譯】[Effective C++第三版?中文版][第20條]盡量使用“引用常量”傳參,而不是傳值  回復  更多評論   

說的很對
2007-06-04 14:02 | picasa
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲午夜激情免费视频| 99国产精品国产精品久久| 欧美日韩不卡| 亚洲欧美清纯在线制服| 欧美色综合网| 国产一区二区欧美| 日韩视频免费在线观看| 久久久免费精品| 久久久一本精品99久久精品66| 欧美777四色影视在线| 国产伦理精品不卡| 亚洲精品视频在线| 免费看亚洲片| 欧美影院视频| 国产午夜精品美女毛片视频| 中日韩午夜理伦电影免费| 欧美成人精品在线播放| 久久成人18免费观看| 国产精品视频免费一区| 亚洲福利国产精品| 久久久久国产精品一区三寸| 亚洲大胆人体在线| 久久久久成人精品| 欧美一区二区三区四区在线| 国产一区二区三区高清在线观看| 亚洲一区二区三区在线视频| 亚洲乱码久久| 国产精品国内视频| 久久99在线观看| 久久精品中文字幕一区| 亚洲福利国产| 日韩亚洲视频| 欧美性猛交xxxx乱大交退制版| 亚洲摸下面视频| 欧美日韩高清在线一区| 亚洲宅男天堂在线观看无病毒| 亚洲精选成人| 国产乱码精品| 亚洲国产精品va在线看黑人| 欧美精品免费观看二区| 亚洲视频一区二区在线观看 | 久久激情五月婷婷| 欧美自拍偷拍午夜视频| 亚洲欧洲精品成人久久奇米网| 日韩小视频在线观看专区| 国产亚洲成精品久久| 欧美国产在线视频| 国产精品男gay被猛男狂揉视频| 久久精品在线观看| 亚洲欧美日韩高清| 亚洲免费中文字幕| 欧美大片一区| 久久蜜桃资源一区二区老牛| 欧美精品二区| 亚洲国产精品视频一区| 国产情人节一区| 中文av字幕一区| 一本久道久久综合中文字幕 | 欧美在线免费| 欧美亚洲视频一区二区| 欧美伦理一区二区| 亚洲日本在线观看| 在线视频亚洲一区| 欧美久久综合| 亚洲国产成人在线播放| 亚洲欧美日韩精品久久久| 亚洲尤物影院| 国产精品视频免费一区| 午夜日韩电影| 免费日韩av片| 亚洲视频大全| 久久午夜电影网| 亚洲精品日韩久久| 久久久久久久综合日本| 亚洲日本va午夜在线电影 | 亚洲精品视频在线观看免费| 国内精品久久久久久久果冻传媒| 欧美一区二区视频97| 葵司免费一区二区三区四区五区| 狠狠久久婷婷| 欧美三级网址| 美女日韩在线中文字幕| 一区二区欧美在线| 欧美一区二区三区久久精品| 在线看欧美视频| 国产精品超碰97尤物18| 久色成人在线| 亚洲在线一区| 亚洲精品影视| 亚洲第一福利在线观看| 久久爱www久久做| av成人激情| 亚洲国产精品成人精品| 国产日韩欧美精品综合| 欧美精品性视频| 欧美a级片一区| 免费91麻豆精品国产自产在线观看| 欧美高清在线播放| 一本大道久久a久久精品综合| 欧美精品麻豆| 男人的天堂亚洲在线| 久久五月激情| 免费在线成人| 久久精品国产99| 欧美中文字幕在线视频| 国产精品99久久久久久有的能看| 亚洲精品久久久久久久久久久久 | 亚洲黄色一区二区三区| 亚洲国产精品第一区二区| 亚洲精品一二三区| 亚洲最新色图| 午夜久久tv| 久久久久久久久久久久久久一区 | 日韩一区二区高清| 一区二区日本视频| 亚洲无毛电影| 久久狠狠亚洲综合| 免费观看不卡av| 国产精品成人aaaaa网站| 国产精品婷婷午夜在线观看| 国语精品一区| 亚洲视频导航| 亚洲电影中文字幕| 亚洲午夜小视频| 久久精品系列| 亚洲第一视频网站| 亚洲免费影视第一页| 欧美高清视频一区| 国产欧美一区二区三区久久| 在线观看不卡av| 午夜精品亚洲| 亚洲日韩中文字幕在线播放| 欧美综合国产| 国产精品久久7| 先锋影音国产一区| 先锋影音久久| 亚洲激情在线| 亚洲国产天堂久久综合| 久久噜噜噜精品国产亚洲综合 | 亚洲欧洲日本国产| 欧美国产一区在线| 中文一区字幕| 亚洲一区二区三区四区视频| 国产欧美一区二区视频| 蜜桃久久精品一区二区| 欧美美女视频| 亚洲国产合集| 中日韩美女免费视频网址在线观看 | 葵司免费一区二区三区四区五区| 久久九九全国免费精品观看| 91久久精品国产91久久性色| 亚洲欧洲在线播放| 国产亚洲精品久久久久动| 亚洲激情六月丁香| 久久99伊人| 亚洲高清一区二区三区| 亚洲在线第一页| 99精品久久| 欧美激情综合在线| 久久久噜噜噜久久| 国产精品九九| 亚洲精品一区在线观看| 亚洲国产片色| 久久xxxx| 亚洲欧美在线aaa| 毛片精品免费在线观看| 久久婷婷蜜乳一本欲蜜臀| 国产精品每日更新| 亚洲深夜福利网站| 亚洲一区二区免费| 欧美视频网址| 亚洲无线一线二线三线区别av| 99re66热这里只有精品3直播| 久久婷婷蜜乳一本欲蜜臀| 久久av资源网站| 伊人夜夜躁av伊人久久| 久久视频一区二区| 欧美国产欧美亚洲国产日韩mv天天看完整 | 欧美大色视频| 日韩视频一区二区三区在线播放免费观看 | 亚洲国产精品一区二区www在线| 韩国v欧美v日本v亚洲v| 久久美女性网| 亚洲高清免费| 亚洲资源av| 黄色成人在线免费| 欧美精品七区| 亚洲在线第一页| 欧美成年人视频网站欧美| 亚洲国产日韩一区| 久久嫩草精品久久久精品| 国产日韩高清一区二区三区在线| 欧美一二三视频| 久久一区激情| 久久国内精品视频| 在线播放日韩欧美| 国产精品theporn| 欧美freesex8一10精品| 亚洲夜晚福利在线观看| 亚洲第一在线|