單參數(shù)構造函數(shù)是一種比較特殊的構造函數(shù)
class A
{
public:
A(int a) //單一參數(shù)構造函數(shù)
{
nMember = a;
}
private:
int nMember;
}
void main()
{
A aInstance = 2; //此處會調(diào)用單一參數(shù)構造函數(shù),而不會調(diào)用賦值運算符,切記:這種調(diào)用方式屬于隱式調(diào)用
}
這種調(diào)用方式,會使得程序不容易正確理解,因此程序員,為了防止這樣的調(diào)用發(fā)生,可以使用關鍵字
explicit修飾單一參數(shù)構造函數(shù),防止其被不可預知的調(diào)用,保證其只能通過顯示調(diào)用的方式調(diào)用。
單參數(shù)構造函數(shù)的調(diào)用時機;
- 用于初始化對象
A aInstance = x; //此時調(diào)用
- 作為函數(shù)參數(shù)被調(diào)用時
void Fun(A cAPar) {...}
Fun(x) //此時d調(diào)用
- 作為函數(shù)返回值僅限于值傳遞
A Get() {
T x;
return x;
} //此時調(diào)用
一個極為明顯的例子是,拷貝構造函數(shù),拷貝構造函數(shù)顯然是一個單一參數(shù)構造函數(shù),它具備單一參數(shù)構造函數(shù)的一切特征。
拷貝構造函數(shù)何時會被調(diào)用:
- 用于初始化對象
A aInstance1 = aInstance2; //此時調(diào)用
- 作為函數(shù)參數(shù)被調(diào)用
void Fun(A cAPar) {...}
Fun(cAinstance) //此時調(diào)用
- 作為函數(shù)返回值僅限于值傳遞
A Get() {
A cAinstance(10);
return cAinstance;
} //此時調(diào)用
用單個實參來調(diào)用的構造函數(shù)定義了從形參類型到類類型的一個隱式轉(zhuǎn)換。
class Sales_item
{
public:
std::istream& input(std::istream& in);
std::ostream& output(std::ostream& out);
inline double avg_price() const;
bool same_isbn(const Sales_item &rbs) const
{
return isbn == rbs.isbn;
}
Sales_item add(Sales_item& other);
Sales_item(const std::string &book = "7115145547"):isbn(book),units_sold(0),revenue(0.0){}
private:
std::string isbn;
unsigned units_sold;
double revenue;
};
這個類的same_isbn函數(shù)的輸入?yún)?shù)是這個類的類型,但是我們卻可以這樣調(diào)用這個函數(shù):
Sales_item trans1;
string null_book = "9-999-99999-9";
trans1.same_isbn(null_book);
這是因為Sales_item的構造函數(shù)可以是帶單個實參的(也可以不帶實參,因為我定義了默認實參7115145547),這時在調(diào)用trans1.same_isbn(null_book);時,就會發(fā)生類型轉(zhuǎn)化:從string轉(zhuǎn)換為Sales_item。建立一個臨時的類的對象(一旦函數(shù)調(diào)用結束,就不能再放這個對象了)。
為了避免這個情況的發(fā)生,可以將類的構造函數(shù)聲明為explicit:
explicit Sales_item(const std::string &book = "7115145547"):isbn(book),units_sold(0),revenue(0.0){}
這樣,前面的默認類型轉(zhuǎn)化就不起作用了。所以對于單形參構造函數(shù),除非有非常明顯的理由讓他發(fā)生隱式類型轉(zhuǎn)換,否者我們應該把它設計為explicit,防止隱式轉(zhuǎn)化的發(fā)生。
當然我們總可以為轉(zhuǎn)化而顯示的使用構造函數(shù):
trans1.same_isbn(Sales_item(null_book));