1. 程式人生 > 其它 >C++ explicit關鍵字詳解

C++ explicit關鍵字詳解

C++中的explicit關鍵字只能用於修飾只有一個引數的類建構函式, 它的作用是表明該建構函式是顯示的, 而非隱式的, 跟它相對應的另一個關鍵字是implicit, 意思是隱藏的,類建構函式預設情況下即宣告為implicit(隱式).
explicit關鍵字只需用於類內的單引數建構函式前面。由於無引數的建構函式和多引數的建構函式總是顯示呼叫,這種情況在建構函式前加explicit無意義。
google的c++規範中提到explicit的優點是可以避免不合時宜的型別變換,缺點無。所以google約定所有單引數的建構函式都必須是顯示的,只有極少數情況下拷貝建構函式可以不宣告稱explicit。例如作為其他類的透明包裝器的類。
effective c++中說:被宣告為explicit的建構函式通常比其non-explicit兄弟更受歡迎。因為它們禁止編譯器執行非預期(往往也不被期望)的型別轉換。除非我有一個好理由允許建構函式被用於隱式型別轉換,否則我會把它宣告為explicit,鼓勵大家遵循相同的政策。

class CxString // 沒有使用explicit關鍵字的類宣告, 即預設為隱式宣告
{
public:
char *_pstr;
int _size;
CxString(int size)
{
_size = size; // string的預設大小
_pstr = malloc(size + 1); // 分配string的記憶體
memset(_pstr, 0, size + 1);
}
CxString(const char *p)
{
int size = strlen(p);
_pstr = malloc(size + 1); // 分配string的記憶體
strcpy(_pstr, p); // 複製字串
_size = strlen(_pstr);
}
// 解構函式這裡不討論, 省略...
};

// 下面是呼叫:
CxString string1(24); // 這樣是OK的, 為CxString預分配24位元組的大小的記憶體
CxString string2 = 10; // 這樣是OK的, 為CxString預分配10位元組的大小的記憶體
CxString string3; // 這樣是不行的, 因為沒有預設建構函式, 錯誤為: “CxString”: 沒有合適的預設建構函式可用
CxString string4("aaaa"); // 這樣是OK的
CxString string5 = "bbb"; // 這樣也是OK的, 呼叫的是CxString(const char *p)
CxString string6 = 'c'; // 這樣也是OK的, 其實呼叫的是CxString(int size), 且size等於'c'的ascii碼
string1 = 2; // 這樣也是OK的, 為CxString預分配2位元組的大小的記憶體
string2 = 3; // 這樣也是OK的, 為CxString預分配3位元組的大小的記憶體
string3 = string1; // 這樣也是OK的, 至少編譯是沒問題的, 但是如果解構函式裡用free釋放_pstr記憶體指標的時候可能會報錯, 完整的程式碼必須過載運算子"=", 並在其中處理記憶體釋放

class CxString // 使用關鍵字explicit的類宣告, 顯示轉換
{
public:
char *_pstr;
int _size;
explicit CxString(int size)
{
_size = size;
// 程式碼同上, 省略...
}
CxString(const char *p)
{
// 程式碼同上, 省略...
}
};

// 下面是呼叫:
CxString string1(24); // 這樣是OK的
CxString string2 = 10; // 這樣是不行的, 因為explicit關鍵字取消了隱式轉換
CxString string3; // 這樣是不行的, 因為沒有預設建構函式
CxString string4("aaaa"); // 這樣是OK的
CxString string5 = "bbb"; // 這樣也是OK的, 呼叫的是CxString(const char *p)
CxString string6 = 'c'; // 這樣是不行的, 其實呼叫的是CxString(int size), 且size等於'c'的ascii碼, 但explicit關鍵字取消了隱式轉換
string1 = 2; // 這樣也是不行的, 因為取消了隱式轉換
string2 = 3; // 這樣也是不行的, 因為取消了隱式轉換
string3 = string1; // 這樣也是不行的, 因為取消了隱式轉換, 除非類實現操作符"="的過載