C++中的explicit關鍵字介紹

轉自:https://www.cnblogs.com/ymy124/p/3632634.html

首先, C++中的explicit關鍵字只能用於修飾只有一個參數的類構造函數, 它的作用是表明該構造函數是顯示的, 而非隱式的, 跟它相對應的另一個關鍵字是implicit, 意思是隱藏的,類構造函數默認情況下即聲明爲implicit(隱式).

那麼顯示聲明的構造函數和隱式聲明的有什麼區別呢? 我們來看下面的例子:

  1. class CxString  // 沒有使用explicit關鍵字的類聲明, 即默認爲隱式聲明  
  2. {  
  3. public:  
  4.     char *_pstr;  
  5.     int _size;  
  6.     CxString(int size)  
  7.     {  
  8.         _size = size;                // string的預設大小  
  9.         _pstr = malloc(size + 1);    // 分配string的內存  
  10.         memset(_pstr, 0, size + 1);  
  11.     }  
  12.     CxString(const char *p)  
  13.     {  
  14.         int size = strlen(p);  
  15.         _pstr = malloc(size + 1);    // 分配string的內存  
  16.         strcpy(_pstr, p);            // 複製字符串  
  17.         _size = strlen(_pstr);  
  18.     }  
  19.     // 析構函數這裏不討論, 省略...  
  20. };  
  21.   
  22.     // 下面是調用:  
  23.   
  24.     CxString string1(24);     // 這樣是OK的, 爲CxString預分配24字節的大小的內存  
  25.     CxString string2 = 10;    // 這樣是OK的, 爲CxString預分配10字節的大小的內存  
  26.     CxString string3;         // 這樣是不行的, 因爲沒有默認構造函數, 錯誤爲: “CxString”: 沒有合適的默認構造函數可用  
  27.     CxString string4("aaaa"); // 這樣是OK的  
  28.     CxString string5 = "bbb"; // 這樣也是OK的, 調用的是CxString(const char *p)  
  29.     CxString string6 = 'c';   // 這樣也是OK的, 其實調用的是CxString(int size), 且size等於'c'的ascii碼  
  30.     string1 = 2;              // 這樣也是OK的, 爲CxString預分配2字節的大小的內存  
  31.     string2 = 3;              // 這樣也是OK的, 爲CxString預分配3字節的大小的內存  
  32.     string3 = string1;        // 這樣也是OK的, 至少編譯是沒問題的, 但是如果析構函數裏用free釋放_pstr內存指針的時候可能會報錯, 完整的代碼必須重載運算符"=", 並在其中處理內存釋放  

上面的代碼中, "CxString string2 = 10;" 這句爲什麼是可以的呢? 在C++中, 如果的構造函數只有一個參數時, 那麼在編譯的時候就會有一個缺省的轉換操作:將該構造函數對應數據類型的數據轉換爲該類對象. 也就是說 "CxString string2 = 10;" 這段代碼, 編譯器自動將整型轉換爲CxString類對象, 實際上等同於下面的操作:

  1. CxString string2(10);  
  2. 或  
  3. CxString temp(10);  
  4. CxString string2 = temp;  

但是, 上面的代碼中的_size代表的是字符串內存分配的大小, 那麼調用的第二句 "CxString string2 = 10;" 和第六句 "CxString string6 = 'c';" 就顯得不倫不類, 而且容易讓人疑惑. 有什麼辦法阻止這種用法呢? 答案就是使用explicit關鍵字. 我們把上面的代碼修改一下, 如下:

  1. class CxString  // 使用關鍵字explicit的類聲明, 顯示轉換  
  2. {  
  3. public:  
  4.     char *_pstr;  
  5.     int _size;  
  6.     explicit CxString(int size)  
  7.     {  
  8.         _size = size;  
  9.         // 代碼同上, 省略...  
  10.     }  
  11.     CxString(const char *p)  
  12.     {  
  13.         // 代碼同上, 省略...  
  14.     }  
  15. };  
  16.   
  17.     // 下面是調用:  
  18.   
  19.     CxString string1(24);     // 這樣是OK的  
  20.     CxString string2 = 10;    // 這樣是不行的, 因爲explicit關鍵字取消了隱式轉換  
  21.     CxString string3;         // 這樣是不行的, 因爲沒有默認構造函數  
  22.     CxString string4("aaaa"); // 這樣是OK的  
  23.     CxString string5 = "bbb"; // 這樣也是OK的, 調用的是CxString(const char *p)  
  24.     CxString string6 = 'c';   // 這樣是不行的, 其實調用的是CxString(int size), 且size等於'c'的ascii碼, 但explicit關鍵字取消了隱式轉換  
  25.     string1 = 2;              // 這樣也是不行的, 因爲取消了隱式轉換  
  26.     string2 = 3;              // 這樣也是不行的, 因爲取消了隱式轉換  
  27.     string3 = string1;        // 這樣也是不行的, 因爲取消了隱式轉換, 除非類實現操作符"="的重載  

explicit關鍵字的作用就是防止類構造函數的隱式自動轉換.

上面也已經說過了, explicit關鍵字只對有一個參數的類構造函數有效, 如果類構造函數參數大於或等於兩個時, 是不會產生隱式轉換的, 所以explicit關鍵字也就無效了. 例如: 

  1. class CxString  // explicit關鍵字在類構造函數參數大於或等於兩個時無效  
  2. {  
  3. public:  
  4.     char *_pstr;  
  5.     int _age;  
  6.     int _size;  
  7.     explicit CxString(int age, int size)  
  8.     {  
  9.         _age = age;  
  10.         _size = size;  
  11.         // 代碼同上, 省略...  
  12.     }  
  13.     CxString(const char *p)  
  14.     {  
  15.         // 代碼同上, 省略...  
  16.     }  
  17. };  
  18.   
  19.     // 這個時候有沒有explicit關鍵字都是一樣的  

但是, 也有一個例外, 就是當除了第一個參數以外的其他參數都有默認值的時候, explicit關鍵字依然有效, 此時, 當調用構造函數時只傳入一個參數, 等效於只有一個參數的類構造函數, 例子如下:

  1. class CxString  // 使用關鍵字explicit聲明  
  2. {  
  3. public:  
  4.     int _age;  
  5.     int _size;  
  6.     explicit CxString(int age, int size = 0)  
  7.     {  
  8.         _age = age;  
  9.         _size = size;  
  10.         // 代碼同上, 省略...  
  11.     }  
  12.     CxString(const char *p)  
  13.     {  
  14.         // 代碼同上, 省略...  
  15.     }  
  16. };  
  17.   
  18.     // 下面是調用:  
  19.   
  20.     CxString string1(24);     // 這樣是OK的  
  21.     CxString string2 = 10;    // 這樣是不行的, 因爲explicit關鍵字取消了隱式轉換  
  22.     CxString string3;         // 這樣是不行的, 因爲沒有默認構造函數  
  23.     string1 = 2;              // 這樣也是不行的, 因爲取消了隱式轉換  
  24.     string2 = 3;              // 這樣也是不行的, 因爲取消了隱式轉換  
  25.     string3 = string1;        // 這樣也是不行的, 因爲取消了隱式轉換, 除非類實現操作符"="的重載  

以上即爲C++ explicit關鍵字的詳細介紹.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章