天天看點

C++ explict關鍵詞詳解

C++中的explicit關鍵字隻能用于修飾隻有一個參數的類構造函數, 它的作用是表明該構造函數是顯式的, 而非隐式的, 跟它相對應的另一個關鍵字是implicit, 意思是隐藏的,類構造函數預設情況

聽得有點雲裡霧裡~~直接上代碼

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記憶體指針的時候可能會報錯, 完整的代碼必須重載運算符"=", 并在其中處理記憶體釋放  
           

"CxString string2 = 10;" 這句為什麼是可以的呢? 在C++中,如果的構造函數隻有一個參數時, 那麼在編譯的時候就會有一個預設的轉換操作:将該構造函數對應資料類型的資料轉換為該類對象. 也就是說 "CxString string2 = 10;" 這段代碼,編譯器自動将整型轉換為CxString類對象。

相當于:

CxString string2(10);  
或  
CxString temp(10);  
CxString string2 = temp; 
           

加上explict後:

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;        // 這樣也是不行的, 因為取消了隐式轉換, 除非類實作操作符"="的重載  
           

第二種初始化方式會産生錯誤~,因為explicit關鍵字的作用就是防止類構造函數的隐式自動轉換.

說明:

  • explicit關鍵字隻對有一個參數的類構造函數有效, 如果類構造函數參數大于或等于兩個時, 是不會産生隐式轉換的, 是以explicit關鍵字也就無效了
  • 當除了第一個參數以外的其他參數都有預設值的時候, explicit關鍵字依然有效

繼續閱讀