標籤:語言 lan pre cpp strong rom string轉換 convert 語法錯誤
在C++程式中很少有人去使用explicit關鍵字,不可否認,在平時的實踐中確實很少能用的上。再說C++的功能強大,往往一個問題可以利用好幾種C++特性去解決。但稍微留心一下就會發現現有的MFC庫或者C++標準庫中的相關類聲明中explicit出現的頻率是很高的。瞭解explicit關鍵字的功能及其使用對於我們閱讀使用庫是很有協助的,而且在編寫自己的代碼時也可以嘗試使用。既然C++語言提供這種特性,我想在有些時候這種特性將會非常有用。按預設規定,只用傳一個參數的建構函式也定義了一個隱式轉換。舉個例子:編寫一個類CExample,其聲明寫在標頭檔Example.h,如下:
1 //Example.h 2 #pragma once 3 class CExample 4 { 5 public: 6 CExample(void); 7 public: 8 ~CExample(void); 9 10 public: 11 int m_iFirst; 12 int m_iSecond; 13 public: 14 CExample(int iFirst, int iSecond = 4); 15 };
類的實現寫在Example.cpp,如下:
//Example.cpp #include "StdAfx.h" #include "Example.h" CExample::CExample(void):m_iFirst(0) { } CExample::~CExample(void) { } CExample::CExample(int iFirst, int iSecond):m_iFirst(iFirst), m_iSecond(iSecond) { }
主函數如下:
1 //TestExplicitKey.cpp 2 ...//其它標頭檔 3 #include "Example.h" 4 int _tmain(int argc, _TCHAR* argv[]) 5 { 6 CExample objOne; //調用沒有參數的建構函式 7 CExample objTwo(12, 12); //調用有兩個參數的建構函式 8 CExample objThree(12); //同上,可以傳一個參數是因為該建構函式的第二個參數有預設值 9 CExample objFour = 12; //執行了隱式轉換,等價於CExample temp(12);objFour(temp);注意這個地方調用了 10 //編譯器為我們提供的預設複製建構函式 11 return 0; 12 } 如果在建構函式聲明中加入關鍵字
explicit,如下explicit CExample(int iFirst, int iSecond = 4);那麼CExample objFour = 12; 這條語句將不能通過編譯。報錯如下:
error C2440: ‘initializing‘ : cannot convert from ‘int‘ to ‘CExample‘ Constructor for class ‘CExample‘ is declared ‘explicit‘
對於某些類型,這一情況非常理想。但在大部分情況中,隱式轉換卻容易導致錯誤(不是語法錯誤,編譯器不會報錯)。隱式轉換總是在我們沒有察覺的情況下悄悄發生,除非有心所為,隱式轉換常常是我們所不希望發生的。通過將建構函式聲明為explicit(顯式)的方式可以抑制隱式轉換。也就是說,explicit建構函式必須顯式調用。
引用一下Bjarne Stroustrup的例子:
1 class String{ 2 explicit String(int n); 3 String(const char *p); 4 }; 5 String s1 = ‘a‘; //錯誤:不能做隱式char->String轉換 6 String s2(10); //可以:調用explicit String(int n); 7 String s3 = String(10);//可以:調用explicit String(int n);再調用預設的複製建構函式 8 String s4 = "Brian"; //可以:隱式轉換調用String(const char *p);再調用預設的複製建構函式 9 String s5("Fawlty"); //可以:正常調用String(const char *p); 10 void f(String); 11 12 String g() 13 { 14 f(10); //錯誤:不能做隱式int->String轉換 15 f("Arthur"); //可以:隱式轉換,等價於f(String("Arthur")); 16 return 10; //同上 17 } 在實際代碼中的東西可不像這種故意造出的例子。 發生隱式轉換,除非有心利用,隱式轉換常常帶來程式邏輯的錯誤,而且這種錯誤一旦發生是很難察覺的。原則上應該在所有的建構函式前加explicit關鍵字,當你有心利用隱式轉換的時候再去解除explicit,這樣可以大大減少錯誤的發生。 轉自http://blog.csdn.net/chollima/article/details/3486230
【轉】C++中的explicit關鍵字