explicit关键字详解(C++ )
一:首先, C++中的explicit關(guān)鍵字只能用于修飾只有一個參數(shù)的類構(gòu)造函數(shù), 它的作用是表明該構(gòu)造函數(shù)是顯示的, 而非隱式的, 跟它相對應(yīng)的另一個關(guān)鍵字是implicit, 意思是隱藏的,類構(gòu)造函數(shù)默認情況下即聲明為implicit(隱式).
class CxString ?// 沒有使用explicit關(guān)鍵字的類聲明, 即默認為隱式聲明 ?
 { ?
 public: ?
 ? ? char *_pstr; ?
 ? ? int _size; ?
 ? ? CxString(int size) ?
 ? ? { ?
 ? ? ? ? _size = size; ? ? ? ? ? ? ? ?// string的預設(shè)大小 ?
 ? ? ? ? _pstr = malloc(size + 1); ? ?// 分配string的內(nèi)存 ?
 ? ? ? ? memset(_pstr, 0, size + 1); ?
 ? ? } ?
 ? ? CxString(const char *p) ?
 ? ? { ?
 ? ? ? ? int size = strlen(p); ?
 ? ? ? ? _pstr = malloc(size + 1); ? ?// 分配string的內(nèi)存 ?
 ? ? ? ? strcpy(_pstr, p); ? ? ? ? ? ?// 復制字符串 ?
 ? ? ? ? _size = strlen(_pstr); ?
 ? ? } ?
 ? ? // 析構(gòu)函數(shù)這里不討論, 省略... ?
 }; ?
 ??
 ? ? // 下面是調(diào)用: ?
 ??
 ? ? CxString string1(24); ? ? // 這樣是OK的, 為CxString預分配24字節(jié)的大小的內(nèi)存 ?
 ? ? CxString string2 = 10; ? ?// 這樣是OK的, 為CxString預分配10字節(jié)的大小的內(nèi)存 ?
 ? ? CxString string3; ? ? ? ? // 這樣是不行的, 因為沒有默認構(gòu)造函數(shù), 錯誤為: “CxString”: 沒有合適的默認構(gòu)造函數(shù)可用 ?
 ? ? CxString string4("aaaa"); // 這樣是OK的 ?
 ? ? CxString string5 = "bbb"; // 這樣也是OK的, 調(diào)用的是CxString(const char *p) ?
 ? ? CxString string6 = 'c'; ? // 這樣也是OK的, 其實調(diào)用的是CxString(int size), 且size等于'c'的ascii碼 ?
 ? ? string1 = 2; ? ? ? ? ? ? ?// 這樣也是OK的, 為CxString預分配2字節(jié)的大小的內(nèi)存 ?
 ? ? string2 = 3; ? ? ? ? ? ? ?// 這樣也是OK的, 為CxString預分配3字節(jié)的大小的內(nèi)存 ?
 ? ? string3 = string1; ? ? ? ?// 這樣也是OK的, 至少編譯是沒問題的, 但是如果析構(gòu)函數(shù)里用free釋放_pstr內(nèi)存指針的時候可能會報錯, 完整的代碼必須重載運算符"=", 并在其中處理內(nèi)存釋放 ?
?
?
?
二:上面的代碼中, "CxString string2 = 10;" 這句為什么是可以的呢? 在C++中, 如果的構(gòu)造函數(shù)只有一個參數(shù)時, 那么在編譯的時候就會有一個缺省的轉(zhuǎn)換操作:將該構(gòu)造函數(shù)對應(yīng)數(shù)據(jù)類型的數(shù)據(jù)轉(zhuǎn)換為該類對象. 也就是說 "CxString string2 = 10;" 這段代碼, 編譯器自動將整型轉(zhuǎn)換為CxString類對象, 實際上等同于下面的操作:
?
CxString string2(10); ?
 或 ?
 CxString temp(10); ?
 CxString string2 = temp;?
?
三:但是, 上面的代碼中的_size代表的是字符串內(nèi)存分配的大小, 那么調(diào)用的第二句 "CxString string2 = 10;" 和第六句 "CxString string6 = 'c';" 就顯得不倫不類, 而且容易讓人疑惑.?有什么辦法阻止這種用法呢? 答案就是使用explicit關(guān)鍵字. 我們把上面的代碼修改一下, 如下:
class CxString ?// 使用關(guān)鍵字explicit的類聲明, 顯示轉(zhuǎn)換 ?
 { ?
 public: ?
 ? ? char *_pstr; ?
 ? ? int _size; ?
 ? ? explicit CxString(int size) ?
 ? ? { ?
 ? ? ? ? _size = size; ?
 ? ? ? ? // 代碼同上, 省略... ?
 ? ? } ?
 ? ? CxString(const char *p) ?
 ? ? { ?
 ? ? ? ? // 代碼同上, 省略... ?
 ? ? } ?
 }; ?
 ??
 ? ? // 下面是調(diào)用: ?
 ??
 ? ? CxString string1(24); ? ? // 這樣是OK的 ?
 ? ? CxString string2 = 10; ? ?// 這樣是不行的, 因為explicit關(guān)鍵字取消了隱式轉(zhuǎn)換 ?
 ? ? CxString string3; ? ? ? ? // 這樣是不行的, 因為沒有默認構(gòu)造函數(shù) ?
 ? ? CxString string4("aaaa"); // 這樣是OK的 ?
 ? ? CxString string5 = "bbb"; // 這樣也是OK的, 調(diào)用的是CxString(const char *p) ?
 ? ? CxString string6 = 'c'; ? // 這樣是不行的, 其實調(diào)用的是CxString(int size), 且size等于'c'的ascii碼, 但explicit關(guān)鍵字取消了隱式轉(zhuǎn)換 ?
 ? ? string1 = 2; ? ? ? ? ? ? ?// 這樣也是不行的, 因為取消了隱式轉(zhuǎn)換 ?
 ? ? string2 = 3; ? ? ? ? ? ? ?// 這樣也是不行的, 因為取消了隱式轉(zhuǎn)換 ?
 ? ? string3 = string1; ? ? ? ?// 這樣也是不行的, 因為取消了隱式轉(zhuǎn)換, 除非類實現(xiàn)操作符"="的重載 ?
?
?
?四:explicit關(guān)鍵字的作用就是防止類構(gòu)造函數(shù)的隱式自動轉(zhuǎn)換.
 上面也已經(jīng)說過了,?explicit關(guān)鍵字只對有一個參數(shù)的類構(gòu)造函數(shù)有效, 如果類構(gòu)造函數(shù)參數(shù)大于或等于兩個時, 是不會產(chǎn)生隱式轉(zhuǎn)換的, 所以explicit關(guān)鍵字也就無效了.?例如:?
?
class CxString ?// explicit關(guān)鍵字在類構(gòu)造函數(shù)參數(shù)大于或等于兩個時無效 ?
 { ?
 public: ?
 ? ? char *_pstr; ?
 ? ? int _age; ?
 ? ? int _size; ?
 ? ? explicit CxString(int age, int size) ?
 ? ? { ?
 ? ? ? ? _age = age; ?
 ? ? ? ? _size = size; ?
 ? ? ? ? // 代碼同上, 省略... ?
 ? ? } ?
 ? ? CxString(const char *p) ?
 ? ? { ?
 ? ? ? ? // 代碼同上, 省略... ?
 ? ? } ?
 }; ?
 ??
 ? ? // 這個時候有沒有explicit關(guān)鍵字都是一樣的 ?
?
?
五:但是, 也有一個例外, 就是當除了第一個參數(shù)以外的其他參數(shù)都有默認值的時候,?explicit關(guān)鍵字依然有效, 此時, 當調(diào)用構(gòu)造函數(shù)時只傳入一個參數(shù),?等效于只有一個參數(shù)的類構(gòu)造函數(shù), 例子如下:
?
class CxString ?// 使用關(guān)鍵字explicit聲明 ?
 { ?
 public: ?
 ? ? int _age; ?
 ? ? int _size; ?
 ? ? explicit CxString(int age, int size = 0) ?
 ? ? { ?
 ? ? ? ? _age = age; ?
 ? ? ? ? _size = size; ?
 ? ? ? ? // 代碼同上, 省略... ?
 ? ? } ?
 ? ? CxString(const char *p) ?
 ? ? { ?
 ? ? ? ? // 代碼同上, 省略... ?
 ? ? } ?
 }; ?
 ??
 ? ? // 下面是調(diào)用: ?
 ??
 ? ? CxString string1(24); ? ? // 這樣是OK的 ?
 ? ? CxString string2 = 10; ? ?// 這樣是不行的, 因為explicit關(guān)鍵字取消了隱式轉(zhuǎn)換 ?
 ? ? CxString string3; ? ? ? ? // 這樣是不行的, 因為沒有默認構(gòu)造函數(shù) ?
 ? ? string1 = 2; ? ? ? ? ? ? ?// 這樣也是不行的, 因為取消了隱式轉(zhuǎn)換 ?
 ? ? string2 = 3; ? ? ? ? ? ? ?// 這樣也是不行的, 因為取消了隱式轉(zhuǎn)換 ?
 ? ? string3 = string1; ? ? ? ?// 這樣也是不行的, 因為取消了隱式轉(zhuǎn)換, 除非類實現(xiàn)操作符"="的重載 ?
?
?
總結(jié)
以上是生活随笔為你收集整理的explicit关键字详解(C++ )的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
 
                            
                        - 上一篇: 《大数据技术原理与应用》第三版 林子雨编
- 下一篇: sql优化【转】
