C&C++ explicit關(guān)鍵字用法

本文根據(jù)眾多互聯(lián)網(wǎng)博客內(nèi)容整理后形成,引用內(nèi)容的版權(quán)歸原始作者所有,僅限于學(xué)習(xí)研究使用,不得用于任何商業(yè)用途。
首先, C++中的explicit關(guān)鍵字只能用于修飾只有一個參數(shù)的類構(gòu)造函數(shù), 它的作用是表明該構(gòu)造函數(shù)是顯示的, 而非隱式的, 跟它相對應(yīng)的另一個關(guān)鍵字是implicit, 意思是隱藏的,類構(gòu)造函數(shù)默認情況下即聲明為implicit(隱式).
那么顯示聲明的構(gòu)造函數(shù)和隱式聲明的有什么區(qū)別呢? 我們來看下面的例子:

class CxString  // 沒有使用explicit關(guān)鍵字的類聲明, 即默認為隱式聲明  
{  
public:  
    char *_pstr;  
    int _size;  
    CxString(int size)  
    {  
        _size = size;                // string的預(yù)設(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);            // 復(fù)制字符串  
        _size = strlen(_pstr);  
    }  
    // 析構(gòu)函數(shù)這里不討論, 省略...  
};  
  
    // 下面是調(diào)用:  
    CxString string1(24);     // 這樣是OK的, 為CxString預(yù)分配24字節(jié)的大小的內(nèi)存  
    CxString string2 = 10;    // 這樣是OK的, 為CxString預(yù)分配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預(yù)分配2字節(jié)的大小的內(nèi)存  
    string2 = 3;              // 這樣也是OK的, 為CxString預(yù)分配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)鍵字都是一樣的  

但是, 也有一個例外, 就是當(dāng)除了第一個參數(shù)以外的其他參數(shù)都有默認值的時候, explicit關(guān)鍵字依然有效, 此時, 當(dāng)調(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)操作符"="的重載  

以上即為C++ explicit關(guān)鍵字的詳細介紹.

參考文檔
C++ explicit關(guān)鍵字詳解

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容