Effective C++筆記之五:了解C++默默編寫并調(diào)用哪些函數(shù)
什么時候empty class(空類)不再是個empty class呢?當C++處理過它之后。是的,如果你自己沒聲明,編譯器就會為它聲明(編譯器版本的)一個copy 構造函數(shù)、一個copy assignment操作符和一個析構函數(shù)。此外如果你沒有聲明任何構造函數(shù),編譯器也會為你聲明一個default構造函數(shù)。所有這些函數(shù)都是public且inline。因此,如果你寫下:
class Empty {};
? ? ? ?這就好像你寫下這樣的代碼:
class?Empty?? {?? public:?? ??Empty(){.....}???????//default構造函數(shù)?? ??~Empty(){.....}??????//析構函數(shù)?? ??Empty(const?Empty?&?rhs)?{.....}??????????????//copy構造函數(shù)?? ??Empty?&?operator=(const?Empty?&?rhs)?{.....}??//copy?assignment操作符?? };
? ? ? ?惟有當這些函數(shù)被需要(被調(diào)用),它們才會被編譯器創(chuàng)建出來。下面代碼造成上述每一個函數(shù)被編譯器產(chǎn)出:
Empty?e1;????//default構造函數(shù) ?????????????//析構函數(shù) Empty?e2(el);//copy構造函數(shù) e2?=?e1;?????//copy?assignment操作符
需要注意的:
1.編譯器產(chǎn)出的析構函數(shù)是個non-virtual,除非這個class 的base class自身聲明有virtual析構函數(shù)(這種情況下這個函數(shù)的虛屬性(virtualness)主要來自base class)。
2.如果手動聲明了一個構造函數(shù),編譯器不再為它創(chuàng)建default構造函數(shù).
3.默認賦值運算符不能生成的3種情況:數(shù)據(jù)成員是引用型、const型或者父類的賦值運算符聲明為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”中不可用 }
? ? ? ?導致error是因為C++并不允許"讓reference改指向不同對象",且更改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”類中聲明) }
? ? ? ?導致error是因為如果某個base classes將copy assignnment操作符聲明為private,編譯器將拒絕為其derived classes生成一個copy assignment操作符。畢竟編譯器為derived classes所生的copy assignment操作符想象中可以處理base class成分,但它們當然無法調(diào)用derived class無權調(diào)用的成員函數(shù),此時編譯器就無能為力了。