const 是constant 的縮寫,「恆定不變」的意思。被const 修飾的東西都受到強制保護,能夠預防意外的變更,能提升程序的健壯性。因此不少C++程序設計書籍建議:「Use const whenever you need」。數組
(1)用函數參數進行返回值的:不用const修飾,不然「指針傳遞/引用傳遞」都不能正確返回參數值;函數
(2)函數參數做輸入:this
例如StringCopy 函數:spa
void StringCopy(char *strDestination, const char *strSource);.net
其中strSource 是輸入參數,strDestination 是輸出參數。給strSource 加上const修飾後,若是函數體內的語句試圖改strSource 的內容,編譯器將指出錯誤。設計
不要將函數void Func1(int x) 寫成void Func1(const int x)指針
不要將函數void Func2(A a) 寫成void Func2(const A a)code
關於內部和非內部數據類型參數傳遞:對象
「const &」修飾輸入參數的用法:blog
(1)若是給以「指針傳遞」方式的函數返回值加const 修飾,那麼函數返回值(即指針)的內容不能被修改,該返回值只能被賦給加const 修飾的同類型指針。
例如函數:const char * GetString(void);
以下語句將出現編譯錯誤:char *str = GetString();
正確的用法是:const char *str = GetString();
(2)若是函數返回值採用「值傳遞方式」,因爲函數會把返回值複製到外部臨時的存儲單元中,加const 修飾沒有任何價值。
例如不要把函數int GetInt(void) 寫成const int GetInt(void)。
同理不要把函數A GetA(void) 寫成const A GetA(void),其中A 爲用戶自定義的數據類型。
函數返回值採用「引用傳遞」的場合並很少,這種方式通常只出如今類的賦值函數中,目的是爲了實現鏈式表達。
示例:
class A { A & operate = (const A &other); // 賦值函數 };
A a, b, c; // a, b, c 爲A 的對象 a = b = c; // 正常的鏈式賦值 (a = b) = c; // 不正常的鏈式賦值,但合法
解答:
任何不會修改數據成員的函數都應該聲明爲const 類型。若是在編寫const 成員函數時,不慎修改了數據成員,或者調用了其它非const 成員函數,編譯器將指出錯誤,這無疑會提升程序的健壯性。
示例:
如下程序中,類stack 的成員函數GetCount 僅用於計數,從邏輯上講GetCount 應當爲const 函數。編譯器將指出GetCount 函數中的錯誤。
class Stack { public: void Push(int elem); int Pop(void); int GetCount(void) const; // const 成員函數 private: int m_num; int m_data[100]; }; int Stack::GetCount(void) const { ++m_num; // 編譯錯誤,企圖修改數據成員m_num Pop(); // 編譯錯誤,企圖調用非const 函數 return m_num; }
const 成員函數的聲明看起來怪怪的:const 關鍵字只能放在函數聲明的尾部,大概是由於其它地方都已經被佔用了。
關於const函數的幾點規則:
const function使用實例:
(1)const 函數只能調用 const 函數,即便某個函數本質上沒有修改任何數據,但沒有聲明爲const,也是不能被const函數調用的。
示例1:
class my { public: string& operator[](const string& s) const { return table[s]; } //編譯錯誤 private: map table; };
解答:
當本重載函數聲明爲const後,就只能調用其餘的一樣也聲明爲const的函數;而table[s]實際上調用的是map的下標重載運算函數,該函數並無聲明爲const。因此去掉本函數的const聲明,就能夠順利經過編譯了。
示例2:
class A{ public: A(int n) { num = n; } void incr() { num += 5; }
void disp() const { cout << numspan> int times(int m) { return num * m; } int f() const { incr();//passing `const A' as `this' argument of `void A::incr()' discards qualifiers disp();// ok times(2);//passing `const A' as `this' argument of `int A::times(int)' discards qualifiers return num; }
private: int num; }; int main(int argc, char *argv[]) { A a(5); a.f(); system("PAUSE"); return EXIT_SUCCESS; }
示例3:
DateTime類是自定義的,isvalid是DateTime類的方法。
bool DateTime::earlier(const DateTime &dt){ if(!isvalid()) return true; if(!dt.isvalid()) return false; //出錯,提示 The non-const member function "DateTime::isvalid()" is called for "const DateTime" }
解答:
若是用const來修飾函數,那麼函數必定是類的成員函數。
const 類型的成員函數不能返回非const類型的引用。
示例:
class Test { public : int & GetValue()const; private: int value; }; int &Test::GetValue() const { return value; //value此時具備const屬性,與返回值類型int &的非const屬性不匹配 }
解答:
修改:
int &Test::GetValue() const { int temp = value; return temp; }
cosnt函數(const放在函數後面):
在調用成員函數時,對象的傳遞是經過this隱藏的,const實際上修飾了this指向的對象,假如定義了:
int &Test::GetValue() const { int temp = value; return temp; }
在C中,常量只的是字面值,如0x1,1L等,const稱爲只讀常量。
const聲明的變量在棧裏面,只是其內容不能在代碼中改變,它不是在靜態內存中的,訪問它的時候彈棧。
訪問cosnt變量時,不能像真正的常量同樣從一個已知的固定內存地址中取得,這對早期的C語言編譯器是不可能的。
因爲const是定義在棧裏面的,它在程序運行時才能肯定裏面的值是多少,因此在編譯的時候不能用它來定義數組大小了 。
轉載自:http://blog.csdn.net/zcf1002797280/article/details/7816977(侵權即刪)