Effective C++學(xué)習(xí)筆記:多才多藝的const,盡可能使用
關(guān)鍵字const多才多藝。你可以用它在classes外部修飾global或namespace(見(jiàn)Effective C++筆記之二) 作用域中的常量,或修飾文件、函數(shù)、或區(qū)塊作用域(block scope)中被聲明為static的對(duì)象。你也可以用它修飾classes內(nèi)部的static和non-static成員變量。面對(duì)指針,你也可以指出指針自身、指針?biāo)肝?,或兩者?或都不)是const:
char?greeting[]="Hello";?? char*?p1=greeting;??????//?non-const?pointer,non-const?data const?char*?p2=greeting;//?non-const?pointer,const?data?? char?const*?p3=greeting;//?和上一行意義相同 char*?const?p4=greeting;//?const?pointer,non-const?data?? const?char*?const?p5?=greeting;//?const?pointer,const?data
一.const作用于迭代器
? ? ? ?STL選代器系以指針為根據(jù)塑模出來(lái),所以迭代器的作用就像個(gè)T*指針。聲明選代器為const就像聲明指針為const一樣(即聲明一個(gè)T* const 指針) ,表示這個(gè)迭代器不得指向不同的東西,但它所指的東西的值是可以改動(dòng)的。如果你希望迭代器所指的東西不可被改動(dòng)(即希望STL模擬一個(gè)const T* 指針) ,你需要
的是const_iterator:
std::vectorvec; ..... const?std::vector::iterator?iter?=?vec.begin(?); *iter?=?10;//?OK ++iter;//?error std::vector::const_iterator?cIter?=?vec.begin(?); *cIter?=?10;//?error ++cIter;//?ok
二.const作用于自定義類型的對(duì)象
? ? ? ?在定義對(duì)象時(shí)指定對(duì)象為常對(duì)象。常對(duì)象中的數(shù)據(jù)成員為常變量,例如:
#includeusing?namespace?std;
class?Time
{
public:
????void?printf()?const
????{
???????//h?=?10;//error?C3490:?由于正在通過(guò)常量對(duì)象訪問(wèn)“h”,因此無(wú)法對(duì)其進(jìn)行修改
???????m?=?10;//?ok
???????cout?<<?"Hour:"?<<?h?<<?"Minute:"?<<?m?<<?"Second:"?<<?s?<<?endl;
????}
????void?show()//?不會(huì)導(dǎo)致編譯錯(cuò)誤
????{
???????h?=?10;
????}
private:
????int?h;
????mutable?int?m;
????int?s;
};
int?main()
{
????const?Time?t;
????t.printf();
????system("pause");
????return?0;
}? ? ? ?
? ? ? ?常對(duì)象t中的數(shù)據(jù)成員雖然未顯示定義為const數(shù)據(jù)成員,但它們都是常變量,無(wú)法修改它們的值。
? ? ? ?常成員函數(shù)printf可以訪問(wèn)常對(duì)象中的數(shù)據(jù)成員,但是不允許修改常對(duì)象中的數(shù)據(jù)成員,除非該數(shù)據(jù)成員被聲明為mutable。
? ? ? ?普通成員函數(shù)show雖然不會(huì)導(dǎo)致編譯錯(cuò)誤,但是無(wú)法被常對(duì)象調(diào)用,因?yàn)槌?duì)象、指向常對(duì)象的指針或引用只能用于調(diào)用其const型成員函數(shù),而不能調(diào)用其非const型的成員函數(shù)。
三.const作用于函數(shù)
1.令函數(shù)返回一個(gè)常量值
? ? ? ?這樣做往往可以降低因客戶錯(cuò)誤而造成的意外,而又不至于放棄安全性和高效性,例如,考慮有理數(shù)(rational numbers)的operator*聲明:
class?Rational{?...?};
const?Rational?operator*?(const?Rational&?lhs,?const?Rational&?rhs);? ? ? ?這個(gè)聲明能很好的杜絕由于筆誤而導(dǎo)致的如下操作:
Rational?a,b,c;
if(a?*?b?=?c)
{
???......
}? ? ? ?如果a和b都是內(nèi)置類型,這樣的代碼直截了當(dāng)就是不合法。而一個(gè)"良好的用戶自定義類型"的特征是它們避免無(wú)端地與內(nèi)置類型不兼容。
2.const參數(shù)
? ? ? ?至于const參數(shù),沒(méi)有什么特別新穎的觀念,它們不過(guò)就像local const對(duì)象一樣,你應(yīng)該在必要使用它們的時(shí)候使用它們。除非你有需要改動(dòng)參數(shù)或local對(duì)象,否則請(qǐng)將它們聲明為const。只不過(guò)多打6個(gè)字符,卻可以省下惱人的錯(cuò)誤,像是"想要鍵入'=='卻意外鍵成'=的錯(cuò)誤,一如稍早所述。
四.const數(shù)據(jù)成員
? ? ? ?常數(shù)據(jù)成員的值是不能改變的,且必須初始化,因?yàn)槌?shù)據(jù)成員是不能被賦值的,關(guān)于初始化,詳見(jiàn):Effective C++筆記之一:聲明、定義、初始化與賦值
#includeusing?namespace?std;
class?Time
{
public:
???void?printf()?const
???{
???????//h?=?10;//error?C3490:?由于正在通過(guò)常量對(duì)象訪問(wèn)“h”,因此無(wú)法對(duì)其進(jìn)行修改
???????m?=?10;
???????cout?<<?"Hour:"?<<?h?<<?"Minute:"?<<?m?<<?"Second:"?<<?s?<<?endl;
???}
???//void?show()//?C2166:?左值指定const對(duì)象
???//{
???// h?=?10;
???//}
private:
???//const?int?h;//?報(bào)錯(cuò),沒(méi)有初始化
???const?int??h?=?10;//?初始化
???mutable?int?m;
???const?int?s?=?10;//?初始化
};
int?main()
{
???const?Time?t;
???t.printf();
???system("pause");
???return?0;
}? ? ??
? ? ? ?對(duì)比“二.const作用于自定義類型的對(duì)象”小節(jié),可看出const對(duì)象中const數(shù)據(jù)成員和非const數(shù)據(jù)成員的區(qū)別。
五.const成員函數(shù)
? ? ? ?常成員函數(shù)只能引用本類中的數(shù)據(jù)成員(const和非const),而不能修改他們。
? ? ? ?const是函數(shù)類型的一部分,在聲明函數(shù)和定義函數(shù)時(shí)都要有const關(guān)鍵字,在調(diào)用時(shí)不必加const。const員函數(shù)可以引用const數(shù)據(jù)成員,也可以引用非const數(shù)據(jù)成員。const數(shù)據(jù)成員可以被const成員函數(shù)引用,也可以被非const數(shù)據(jù)成員函數(shù)引用,但是不能被修改。
需要注意的是:
1.不要誤認(rèn)為常對(duì)象中的成員函數(shù)都是常成員函數(shù)。常對(duì)象只能保證所有的數(shù)據(jù)成員的值不被修改。如果在對(duì)象中的成員函數(shù)為加const聲明,編譯系統(tǒng)把它作為非const成員函數(shù)處理。
2.兩個(gè)成員函數(shù)如果只是常量性(constness)不同,可以被重載。舉個(gè)例子
#includeusing?namespace?std;
class?TextBlock
{
public:
???TextBlock::TextBlock(string?str)?:test(str)
???{
???}
???const?char?&?operator[](size_t?position)?const
???{
return?test[position];
???}
???char?&?operator[](size_t?position)?
???{
return?test[position];
???}
private:
string?test;
};
int?main()
{
???TextBlock?tb("Hello");
???cout?<<?tb[0]?<<?endl;
???const?TextBlock?ctb("World");
???cout?<<?ctb[0]?<<?endl;
???system("pause");
???return?0;
}3.常成員函數(shù)不能調(diào)用另一個(gè)非const成員函數(shù)。但是非const成員函數(shù)可以調(diào)用const成員函數(shù)。舉個(gè)例子:
在上述例子中,重載的兩個(gè)操作符函數(shù)體內(nèi)的代碼量小,感覺(jué)不到有什么不妥。但是如果代碼量大的話,可以讓非const operator[]調(diào)用其const兄弟來(lái)避免代碼重復(fù)。
#includeusing?namespace?std;
class?TextBlock
{
public:
???TextBlock::TextBlock(string?str)?:test(str)
???{
???}
???const?char?&?operator[](size_t?position)?const
???{
return?test[position];
???}
???char?&?operator[](size_t?position)?//?先調(diào)用const版本的操作符,然后去掉返回結(jié)果的const屬性
???{
return
????const_cast(
static_cast(*this)[position]);
???}
private:
string?test;
};
int?main()
{
???TextBlock?tb("Hello");
???cout?<<?tb[0]?<<?endl;
???const?TextBlock?ctb("World");
???cout?<<?ctb[0]?<<?endl;
???system("pause");
???return?0;
}六.指向?qū)ο蟮某V羔?br />? ? ? ?將指向?qū)ο蟮闹羔樧兞柯暶鳛閏onst型并將之初始化,這樣指針值始終保持為其初始值,不能改變,即其指向始終不變。如:
int?a?=?1?,?b; int?*?const?ptr1?=?&a; ptr1?=?&b;//?error?C3892:?“ptr1”:?不能給常量賦值
? ? ? ?指向?qū)ο蟮某V羔樀闹挡荒芨淖?,即始終指向同一個(gè)對(duì)象,但可以改變其所指向?qū)ο螅ㄈ鏱)中數(shù)據(jù)成員的值。
七.指向常對(duì)象的指針
1.如果一個(gè)變量已經(jīng)被聲明為常變量,只能用指向常變量的指針指向它,而不能用一般的(指向那個(gè)非const型變量的)指針變量去指向它。
2.指向常變量的指針除了可以指向常變量,還可以指向非const變量。此時(shí)不能通過(guò)指針變量改變?cè)撟兞康闹怠?br />
3.指向常對(duì)象的指針常作為函數(shù)參數(shù)。
八.對(duì)象的常引用
? ? ? ?在C++面向?qū)ο蟪绦蛟O(shè)計(jì)中,經(jīng)常用常指針和常引用作函數(shù)參數(shù)。這樣既能保證數(shù)據(jù)安全,使數(shù)據(jù)不能被隨意修改,在調(diào)用函數(shù)時(shí)又不必建立實(shí)參的拷貝。





