亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

Effective C++ 2e Item45

2019-09-10 09:07:16
字體:
來源:轉載
供稿:網友

雜項

進行高效的C++程序設計有很多準則,其中有一些很難歸類。本章就是專門為這些準則而安排的。不要因此而小看了它們的重要性。要想寫出高效的軟件,就必須知道:編譯器在背后為你(給你?)做了些什么,怎樣保證非局部的靜態對象在被使用前已經被初始化,能從標準庫得到些什么,從何處著手深入理解語言底層的設計思想。本書最后的這個章節,我將詳細說明這些問題,甚至更多其它問題。


條款45: 弄清C++在幕后為你所寫、所調用的函數

一個空類什么時候不是空類? ---- 當C++編譯器通過它的時候。如果你沒有聲明下列函數,體貼的編譯器會聲明它自己的版本。這些函數是:一個拷貝構造函數,一個賦值運算符,一個析構函數,一對取址運算符。另外,如果你沒有聲明任何構造函數,它也將為你聲明一個缺省構造函數。所有這些函數都是公有的。換句話說,如果你這么寫:

class Empty{};

和你這么寫是一樣的:

class Empty {
public:
 Empty();/t/t        // 缺省構造函數
 Empty(const Empty& rhs);        // 拷貝構造函數

 ~Empty();/t/t       // 析構函數 ---- 是否
/t/t/t/t  // 為虛函數看下文說明
 Empty&
 operator=(const Empty& rhs);    // 賦值運算符

 Empty* operator&();/t     // 取址運算符
 const Empty* operator&() const;
};

現在,如果需要,這些函數就會被生成,但你會很容易就需要它們。下面的代碼將使得每個函數被生成:

const Empty e1;/t/t     // 缺省構造函數
/t/t/t/t    // 析構函數

Empty e2(e1);/t/t       // 拷貝構造函數

e2 = e1;/t/t/t    //  賦值運算符

Empty *pe2 = &e2;/t/t   // 取址運算符
/t/t/t/t    // (非const)

const Empty *pe1 = &e1;/t     //  取址運算符
/t/t/t/t    // (const)

假設編譯器為你寫了函數,這些函數又做些什么呢?是這樣的,缺省構造函數和析構函數實際上什么也不做,它們只是讓你能夠創建和銷毀類的對象(對編譯器來說,將一些 "幕后" 行為的代碼放在此處也很方便 ---- 參見條款33和M24。)。注意,生成的析構函數一般是非虛擬的(參見條款14),除非它所在的類是從一個聲明了虛析構函數的基類繼承而來。缺省取址運算符只是返回對象的地址。這些函數實際上就如同下面所定義的那樣:

inline Empty::Empty() {}

inline Empty::~Empty() {}

inline Empty * Empty::operator&() { return this; }

inline const Empty * Empty::operator&() const
{ return this; }

至于拷貝構造函數和賦值運算符,官方的規則是:缺省拷貝構造函數(賦值運算符)對類的非靜態數據成員進行 "以成員為單位的" 逐一拷貝構造(賦值)。即,如果m是類C中類型為T的非靜態數據成員,并且C沒有聲明拷貝構造函數(賦值運算符),m將會通過類型T的拷貝構造函數(賦值運算符)被拷貝構造(賦值)---- 如果T有拷貝構造函數(賦值運算符)的話。如果沒有,規則遞歸應用到m的數據成員,直至找到一個拷貝構造函數(賦值運算符)或固定類型(例如,int,double,指針,等)為止。默認情況下,固定類型的對象拷貝構造(賦值)時是從源對象到目標對象的 "逐位" 拷貝。對于從別的類繼承而來的類來說,這條規則適用于繼承層次結構中的每一層,所以,用戶自定義的構造函數和賦值運算符無論在哪一層被聲明,都會被調用。

我希望這已經說得很清楚了。

但怕萬一沒說清楚,還是給個例子??催@樣一個NamedObject模板的定義,它的實例是可以將名字和對象聯系起來的類:

template<class T>
class NamedObject {
public:
 NamedObject(const char *name, const T& value);
 NamedObject(const string& name, const T& value);

 ...

private:
 string nameValue;
 T objectValue;
};

因為NamedObject類聲明了至少一個構造函數,編譯器將不會生成缺省構造函數;但因為沒有聲明拷貝構造函數和賦值運算符,編譯器將生成這些函數(如果需要的話)。

看下面對拷貝構造函數的調用:

NamedObject<int> no1("Smallest Prime Number", 2);

NamedObject<int> no2(no1);      // 調用拷貝構造函數

編譯器生成的拷貝構造函數必須分別用no1.nameValue和no1.objectValue來初始化no2.nameValue和no2.objectValue。nameValue的類型是string,string有一個拷貝構造函數(你可以在標準庫中查看string來證實 ---- 參見條款49),所以no2.nameValue初始化時將調用string的拷貝構造函數,參數為no1.nameValue。另一方面,NamedObject<int>::objectValue的類型是int(因為這個模板實例中,T是int),int沒有定義拷貝構造函數,所以no2.objectValue是通過從no1.objectValue拷貝每一個比特(bit)而被初始化的。

編譯器為NamedObject<int>生成的賦值運算符也以同樣的方式工作,但通常,編譯器生成的賦值運算符要想如上面所描述的那樣工作,與此相關的所有代碼必須合法且行為上要合理。如果這兩個條件中有一個不成立,編譯器將拒絕為你的類生成operator=,你就會在編譯時收到一些診斷信息。

例如,假設NamedObject象這樣定義,nameValue是一個string的引用,objectValue是一個const T:

template<class T>
class NamedObject {
public:
 // 這個構造函數不再有一個const名字參數,因為nameValue
 // 現在是一個非const string的引用。char*構造函數
 // 也不見了,因為引用要指向的是string
 NamedObject(string& name, const T& value);

 .../t/t/t  // 同上,假設沒有
/t/t/t       // 聲明operator=
private:
 string& nameValue;/t   // 現在是一個引用
 const T objectValue;/t // 現在為const
};

現在看看下面將會發生什么:

string newDog("Persephone");
string oldDog("Satch");

NamedObject<int> p(newDog, 2);      // 正在我寫本書時,我們的
/t/t/t/t    // 愛犬Persephone即將過
/t/t/t/t    // 她的第二個生日

NamedObject<int> s(oldDog, 29);     // 家犬Satch如果還活著,
/t/t/t/t    // 會有29歲了(從我童年時算起)

p = s;/t/t/t      // p中的數據成員將會發生
/t/t/t/t    // 些什么呢?

賦值之前,p.nameValue指向某個string對象,s.nameValue也指向一個string,但并非同一個。賦值會給p.nameValue帶來怎樣的影響呢?賦值之后,p.nameValue應該指向 "被s.nameValue所指向的string" 嗎,即,引用本身應該被修改嗎?如果是這樣,那太陽從西邊出來了,因為C++沒有辦法讓一個引用指向另一個不同的對象(參見條款M1)。或者,p.nameValue所指的string對象應該被修改嗎? 這樣的話,含有 "指向那個string的指針或引用" 的其它對象也會受影響,也就是說,和賦值沒有直接關系的其它對象也會受影響。這是編譯器生成的賦值運算符應該做的嗎?

面對這樣的難題,C++拒絕編譯這段代碼。如果想讓一個包含引用成員的類支持賦值,你就得自己定義賦值運算符。對于包含const成員的類(例如上面被修改的類中的objectValue)來說,編譯器的處理也相似;因為修改const成員是不合法的,所以編譯器在隱式生成賦值函數時也會不知道怎么辦。還有,如果派生類的基類將標準賦值運算符聲明為private,  編譯器也將拒絕為這個派生類生成賦值運算符。因為,編譯器為派生類生成的賦值運算符也應該處理基類部分(見條款16和M33),但這樣做的話,就得調用對派生類來說無權訪問的基類成員函數,這當然是不可能的。

以上關于編譯器生成函數的討論引發了這樣的問題:如果想禁止使用這些函數,那該怎么辦呢?也就是說,假如你永遠不想讓類的對象進行賦值,所以有意不聲明operator=,那該怎么做呢?這個小難題的解決方案正是條款27討論的主題。指針成員和編譯器生成的拷貝構造函數及賦值運算符之間的相互影響經常被人忽視,關于這個話題的討論請查看條款11。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲第五色综合网| 亚洲欧美另类在线观看| 国产丝袜一区视频在线观看| 欧美日韩美女在线| 欧美一级黄色网| 亚洲福利视频久久| 狠狠干狠狠久久| 国产成人精品999| 奇门遁甲1982国语版免费观看高清| 久久久精品免费视频| 国产精品观看在线亚洲人成网| www日韩中文字幕在线看| 日韩欧美在线视频| www日韩中文字幕在线看| 欧美视频13p| 成人美女免费网站视频| 久久中国妇女中文字幕| 欧美日韩国产va另类| 日韩三级成人av网| 久久久在线观看| 久久精品99久久久久久久久| 高清欧美性猛交| 国产综合在线视频| 亚洲护士老师的毛茸茸最新章节| 一区二区三区四区精品| 91在线免费看网站| 欧美精品一区三区| 国产精品对白刺激| 国产精品欧美亚洲777777| 亚洲欧美日韩一区二区在线| 在线视频中文亚洲| 97久久精品人人澡人人爽缅北| 日韩免费av在线| 日韩av在线网页| 自拍偷拍亚洲在线| 欧美大人香蕉在线| 国产亚洲欧美日韩美女| 国产精品久久久久久久久久久久久| 久久躁日日躁aaaaxxxx| 国产精品久久久久久久久久免费| 久久男人av资源网站| 欧美性生交大片免网| 国模视频一区二区三区| 日韩av免费在线观看| 国产精品福利网| 国产精品久久在线观看| 欧美电影《睫毛膏》| 91情侣偷在线精品国产| 精品日本美女福利在线观看| 精品少妇一区二区30p| 久久91精品国产91久久跳| 亚洲国产精品yw在线观看| 国产男女猛烈无遮挡91| 91超碰caoporn97人人| 尤物yw午夜国产精品视频| 国产精品福利片| 日本免费久久高清视频| 亚洲专区国产精品| 国产精品jizz在线观看麻豆| 热久久99这里有精品| 精品人伦一区二区三区蜜桃免费| 精品香蕉一区二区三区| 亚洲激情在线观看| 欧美极品少妇xxxxⅹ免费视频| 精品成人av一区| 亚洲欧美精品中文字幕在线| 欧美精品久久久久久久久| 欧美亚洲一区在线| 欧美成人sm免费视频| 在线播放国产一区二区三区| 亚洲欧美制服另类日韩| 国产97色在线|日韩| 日韩精品免费在线观看| 亚洲欧美日韩精品久久亚洲区| 欧美日韩美女在线| 麻豆国产精品va在线观看不卡| 97不卡在线视频| 亚洲香蕉在线观看| 激情久久av一区av二区av三区| 7777免费精品视频| 日本视频久久久| 97国产精品人人爽人人做| 色999日韩欧美国产| 91高清视频在线免费观看| 欧美整片在线观看| 精品无码久久久久久国产| 欧美日韩国产综合新一区| 在线看片第一页欧美| 亚州成人av在线| 日韩久久精品电影| 久久全球大尺度高清视频| 91中文在线观看| 91在线观看免费网站| 91久久国产综合久久91精品网站| 成人精品一区二区三区电影免费| 国产综合视频在线观看| 97免费中文视频在线观看| 亚洲a∨日韩av高清在线观看| 亚洲第一区中文99精品| 日韩精品欧美激情| 81精品国产乱码久久久久久| 色偷偷9999www| 成人h片在线播放免费网站| 久久在线免费观看视频| 欧美激情亚洲综合一区| 国产精品久久久av久久久| 91最新在线免费观看| 日韩高清不卡av| 国产精品视频网址| 91亚洲一区精品| 亚洲自拍偷拍网址| 性金发美女69hd大尺寸| 欧洲亚洲免费在线| 日韩成人在线网站| 欧美疯狂做受xxxx高潮| 一区二区福利视频| 久久99国产综合精品女同| 成人xxxxx| 在线观看精品自拍私拍| 欧美劲爆第一页| 国产一区二中文字幕在线看| 日韩男女性生活视频| 久久久成人的性感天堂| 久久久久久av| 亚洲qvod图片区电影| 国产在线观看一区二区三区| 亚洲区一区二区| 91精品国产乱码久久久久久蜜臀| 欧美成人手机在线| 国产精品网站视频| 日韩电影网在线| 7777免费精品视频| 亚洲欧洲第一视频| 国产精品免费看久久久香蕉| 欧美激情a∨在线视频播放| 日韩色av导航| 亚洲人成网在线播放| 中文字幕亚洲一区二区三区五十路| 疯狂做受xxxx高潮欧美日本| 亚洲人成网站在线播| 97色在线观看免费视频| 欧美亚洲另类制服自拍| 91国语精品自产拍在线观看性色| 国产精品91久久| 午夜精品一区二区三区视频免费看| 欧美一级片在线播放| 尤物99国产成人精品视频| 欧美极品少妇xxxxⅹ免费视频| 欧美性猛交xxxxx水多| 日韩在线观看精品| 欧美视频在线看| 国产精品久久久久久婷婷天堂| 国产成人福利网站| 26uuu久久噜噜噜噜| 国产有码一区二区| 97在线观看免费高清| 米奇精品一区二区三区在线观看| 欧美肥老太性生活视频| 亚洲国产精品va在线看黑人| 亚洲v日韩v综合v精品v| 91国产中文字幕| 亚洲黄色av女优在线观看| 亚洲在线视频观看| 欧美亚洲国产另类|