Effective C++讀書筆記之二 :盡可能使用const
條款三:盡可能使用const(use const whenever possible)
const允許你指定一個(gè)語義約束,而編譯器會(huì)強(qiáng)制實(shí)施這項(xiàng)約束。它允許你告訴編譯器和其他程序員某值應(yīng)該保持不變。有一條約束需要注意,那就是:如果const出現(xiàn)在*號(hào)的左邊,那就是說被指物是常量;如果出現(xiàn)在星號(hào)右邊,則表示指針本身是常量;如果出現(xiàn)在兩邊,則被指物和指針都是常量。如果被指物是常量,則關(guān)鍵字const寫在類型的前面和類型之后,星號(hào)之前兩種所表示的語義是相同的。例如下面這兩種寫法是一樣的:
void f1(const Widget* pw);
void f2(Widget const * pw);
const也可用來修飾STL中的迭代器。聲明迭代器為const就想聲明指針為const一樣(即T* const 指針),表示這個(gè)迭代器不得指向不同的東西。但它所指的東西的值是可以改變的。如果希望迭代器所指的東西不可改變(即模擬一個(gè)const T*指針),需要的是const_iterator:
std::vector<int> vec;
...
const std::vector<int>::iterator iter = vec.begin();// same as T* const
*iter = 10; //no problem
++iter; //wrong!!
std::vector<int>::const_iterator cIter = vec.begin();//same as const T*
*iter = 10; //wrong!!
++iter; //no problem
const 最具威力(?)的用法是面對函數(shù)聲明時(shí)的應(yīng)用。在一個(gè)函數(shù)聲明式內(nèi),const可以和函數(shù)返回值,各參數(shù),函數(shù)自身(成員函數(shù))產(chǎn)生關(guān)聯(lián)。
令函數(shù)返回一個(gè)常量值,往往可以降低因客戶錯(cuò)誤而造成的意外,而又不至于放棄安全性和高效性。例如,考慮有理數(shù)的operator*聲明式:
class Rational(){...};
const Rational operator* (const Rational & lhs, const Rational & rhs);
也許你會(huì)說為什么返回一個(gè)const對象?原因是如果不這樣別人可能實(shí)現(xiàn)這樣的暴行:
Rational a,b,c;
...
(a*b)=c;
下面,主要說明const作用于成員函數(shù)。
許多人都忽視了這么一個(gè)事實(shí),那就是如果兩個(gè)成員函數(shù)只是常量性不同,那么他們是可以重載的。考慮以下這個(gè)用來表示一大塊文字的class:
























真是情形中const對象多用于passed by pointer-to-const或passed by reference-to-const的傳遞結(jié)果。上述的ctb太過于造作,下邊這個(gè)比較真實(shí):








只用重載operator[]并對不同的版本給予不同的返回類型,就可以令const和non-const獲得不同的處理。
此處需要注意一點(diǎn),non-const operator[]的返回類型是個(gè)reference to char,不是char。如果operator[]返回的是個(gè)char,下邊的賦值就不能通過編譯:
tb[0] = 'x'; //error c2106: ' = ' : left operand must be l-value
那是因?yàn)椋绻瘮?shù)的返回類型是個(gè)內(nèi)置類型,那么改動(dòng)函數(shù)的返回值從來就不合法。縱使合法,C++以by value返回對象這一事實(shí)(條款20)意味著改動(dòng)的其實(shí)只是tb.text[0]的一個(gè)副本,不是tb.text[0]本身,那不是我們想要的結(jié)果。
下邊來說說在const和non-const成員函數(shù)中避免重復(fù)
假設(shè)TextBlock(和CTextBlock)內(nèi)的operator[]不單只是返回一個(gè)reference指向某字符,也執(zhí)行邊界檢查、志記訪問信息、甚至可能進(jìn)行數(shù)據(jù)完整性檢驗(yàn)。把所有這些同時(shí)放進(jìn)const和non-const operator[]中,導(dǎo)致這樣的一個(gè)怪物:

































我們真正要做的,是實(shí)現(xiàn)operator[]的機(jī)能一次并使用它兩次。也就是說,你必須使一個(gè)調(diào)用另一個(gè)。這促使我們將常量性轉(zhuǎn)除(casting away constness)。
就一般而言,casting是一個(gè)糟糕的想法,在條款27中有詳細(xì)的說明。然而代碼重復(fù)也不是什么令人愉快的經(jīng)驗(yàn)。本例中cosnt operator[]完全做掉了non-const版本該做的一切,唯一不同是其返回類型多了一個(gè)const資格修飾。這種情況下如果將返回值的const轉(zhuǎn)除是安全的,因?yàn)椴徽撜l調(diào)用non-const operator[]都一定首先有個(gè)non-const對象,否則就不能夠調(diào)用non-const函數(shù)。所以令non-const operator[]調(diào)用其const兄弟是一個(gè)避免重復(fù)的安全做法:




























下面來考慮一下反向的做法:令const來調(diào)用non-const以避免重復(fù)。這個(gè)不是我們應(yīng)該做的。const成員函數(shù)承諾絕對不改變其對象的邏輯狀態(tài),non-const成員函數(shù)卻沒有這般承諾。如果在const函數(shù)內(nèi)部調(diào)用了non-const函數(shù),就是冒了這樣的風(fēng)險(xiǎn):你曾經(jīng)承諾不改動(dòng)的那個(gè)對象被改動(dòng)了。這就是為什么“const成員函數(shù)調(diào)用non-const成員函數(shù)”是一種錯(cuò)誤行為:因?yàn)閷ο笥锌赡芤虼硕桓膭?dòng)。反向調(diào)用才是安全的:non-const函數(shù)本來就可以對其對象做任何動(dòng)作,所以在其中調(diào)用一個(gè)const成員函數(shù)并不會(huì)帶來任何風(fēng)險(xiǎn)。
本條目總結(jié):
Things to Remember
-
Declaring something const helps compilers detect usage errors. const can be applied to objects at any scope, to function parameters and return types, and to member functions as a whole.
-
Compilers enforce bitwise constness, but you should program using conceptual constness.
-
When const and non-const member functions have essentially identical implementations, code duplication can be avoided by having the non-const version call the const version.