Effective C++筆記之五:了解C++默默編寫并調(diào)用哪些函數(shù)
什么時(shí)候empty class(空類)不再是個(gè)empty class呢?當(dāng)C++處理過它之后。是的,如果你自己沒聲明,編譯器就會(huì)為它聲明(編譯器版本的)一個(gè)copy 構(gòu)造函數(shù)、一個(gè)copy assignment操作符和一個(gè)析構(gòu)函數(shù)。此外如果你沒有聲明任何構(gòu)造函數(shù),編譯器也會(huì)為你聲明一個(gè)default構(gòu)造函數(shù)。所有這些函數(shù)都是public且inline。因此,如果你寫下:
class Empty {};
? ? ? ?這就好像你寫下這樣的代碼:
class?Empty?? {?? public:?? ??Empty(){.....}???????//default構(gòu)造函數(shù)?? ??~Empty(){.....}??????//析構(gòu)函數(shù)?? ??Empty(const?Empty?&?rhs)?{.....}??????????????//copy構(gòu)造函數(shù)?? ??Empty?&?operator=(const?Empty?&?rhs)?{.....}??//copy?assignment操作符?? };
? ? ? ?惟有當(dāng)這些函數(shù)被需要(被調(diào)用),它們才會(huì)被編譯器創(chuàng)建出來。下面代碼造成上述每一個(gè)函數(shù)被編譯器產(chǎn)出:
Empty?e1;????//default構(gòu)造函數(shù) ?????????????//析構(gòu)函數(shù) Empty?e2(el);//copy構(gòu)造函數(shù) e2?=?e1;?????//copy?assignment操作符
需要注意的:
1.編譯器產(chǎn)出的析構(gòu)函數(shù)是個(gè)non-virtual,除非這個(gè)class 的base class自身聲明有virtual析構(gòu)函數(shù)(這種情況下這個(gè)函數(shù)的虛屬性(virtualness)主要來自base class)。
2.如果手動(dòng)聲明了一個(gè)構(gòu)造函數(shù),編譯器不再為它創(chuàng)建default構(gòu)造函數(shù).
3.默認(rèn)賦值運(yùn)算符不能生成的3種情況:數(shù)據(jù)成員是引用型、const型或者父類的賦值運(yùn)算符聲明為private。例如:
#include#includeclass?Dog? { public: Dog(const?std::string&?name,?const?int?age) { this->name?=?name; this->age?=?age; } ~Dog(){?} private: std::string&?name; const?int?age; }; void?main() { std::string?name1("Big?Yellow"); std::string?name2("Big?Yellow"); Dog?bigYellow1(name1,?3);?//?大黃1于去年歸西了 Dog?bigYellow2(name2,?4); bigYellow1?=?bigYellow2;??//?大黃2取代了大黃1的位置 ??????????????????????????//?error?C2582:?“operator?=”函數(shù)在“Dog”中不可用 }
? ? ? ?導(dǎo)致error是因?yàn)镃++并不允許"讓reference改指向不同對(duì)象",且更改const成員是不合法的。? ? ? ? ? ? ? ? ? ? ? ? ?
#include#includeclass?Animal { public: ~Animal(){}; private: Animal?&?operator=(const?Animal?&?rhs);?//copy?assignment?操作符? }; class?Dog?:public?Animal? { public: Dog(const?std::string&?name,?const?int?age) { this->name?=?name; this->age?=?age; } ~Dog(){?} private: std::string?name; int?age; }; void?main() { std::string?name1("Big?Yellow"); std::string?name2("Big?Yellow"); Dog?bigYellow1(name1,?3);?//?大黃1于去年歸西了 Dog?bigYellow2(name2,?4); bigYellow1?=?bigYellow2;??//?大黃2取代了大黃1的位置 ??????????????????????????//?error?C2248:?“Animal::operator?=”:?無法訪問?private?成員(在“Animal”類中聲明) }
? ? ? ?導(dǎo)致error是因?yàn)槿绻硞€(gè)base classes將copy assignnment操作符聲明為private,編譯器將拒絕為其derived classes生成一個(gè)copy assignment操作符。畢竟編譯器為derived classes所生的copy assignment操作符想象中可以處理base class成分,但它們當(dāng)然無法調(diào)用derived class無權(quán)調(diào)用的成員函數(shù),此時(shí)編譯器就無能為力了。