看《C++必知必會(huì)》比看《C++編程思想》編程思想有趣多了,《C++編程思想》是為C程序員寫的,而我基本上一來就直接學(xué)的C++,因而《C++編程思想》并不適合我。而《C++必知必會(huì)》是直接從實(shí)際出發(fā),直接點(diǎn)出來在編程中遇到的各種問題,因而很是實(shí)用,深得我心。
1.?dāng)?shù)據(jù)結(jié)構(gòu)
摘要:為類型選擇一個(gè)描述性名字。如果難以為這個(gè)類型命名,那就說明你還不知道你想要實(shí)現(xiàn)什么。
評(píng):很是精辟,每當(dāng)我想到了這個(gè)類的名字,我就知道了如何去做和做些什么。摘要:列舉類型所能執(zhí)行的操作。......要避免在實(shí)現(xiàn)時(shí)簡(jiǎn)單地為數(shù)據(jù)成員提供一串get/set操作——那不叫做數(shù)據(jù)抽象而是懶惰且缺乏想象力的表現(xiàn)。
評(píng):不太明白,在我看來,C++中的類型僅僅是把數(shù)據(jù)按照一定的規(guī)律(類型)進(jìn)行封裝,其目的就是為了進(jìn)行數(shù)據(jù)之間的交換。我的確不僅僅是用了一串get/set操作,但不是的操作也僅僅比get/set有一點(diǎn)延伸,比如按照某個(gè)條件來get/set一個(gè)或一組數(shù)據(jù)。
2.多態(tài)
摘要:基類可以不知道除自身以外的任何事物。
評(píng):很是恰當(dāng),基類可以看成派生類的接口,派生類可以看成是基類的實(shí)現(xiàn)。
3.設(shè)計(jì)模式
摘要:一旦設(shè)計(jì)完成后,甚至你的經(jīng)理都能夠理解完整的設(shè)計(jì)方案,只要他具備一些必需的模式方面的知識(shí)。
評(píng):以前我一直沒有看過關(guān)于設(shè)計(jì)模式方面的書,也不理解什么是設(shè)計(jì)模式和為什么需要設(shè)計(jì)模式。現(xiàn)在,通俗的話說設(shè)計(jì)模式是程序員之間的“黑話”,它與使用的平臺(tái)、語言毫無關(guān)系,它指的是你在制作程序時(shí)的一種構(gòu)想或方法,而這些構(gòu)造方法都是大家熟知的、經(jīng)過驗(yàn)證的、有效的、高效率的。
4.STL
摘要:STL包含三大組件:容器、算法和迭代器。......STL的優(yōu)秀思想體現(xiàn)在:容器與在容器上執(zhí)行的算法之間無需彼此了解,這種戲法是通過迭代器實(shí)現(xiàn)的。
評(píng):上面已經(jīng)很好的解釋了什么是STL了,而且還道出了C++中的精髓,無需彼此了解,不是侵入式的設(shè)計(jì),還象那個(gè)基類、派生類。
5.引用
評(píng):我只會(huì)把引用用在函數(shù)形參上,一般還給它加上一個(gè)const,其他的地方么,我一般使用指針。
6.?dāng)?shù)組形參
摘要:數(shù)組在傳入時(shí),實(shí)質(zhì)上只傳入指向其首元素的指針。
評(píng):要小心!書中用一個(gè)準(zhǔn)確的詞“退化“來描述這種狀況。不過自從我使用了vector后也就找不出什么理由使用數(shù)組了。我很高興是這樣的結(jié)果。什么,你說你一定要用多維數(shù)組,那么我說,你去死吧,我一般用一維代替多維,或者使用vector進(jìn)行鑲套。
7.產(chǎn)量指針和指向產(chǎn)量的指針
評(píng):我背下了一個(gè)簡(jiǎn)單的方法來區(qū)別這兩個(gè)。我把*和它前面的作為一組,*后面的作為一組,如const T *ptr,讀后面的部分,ptr是指針,指向const T,又如T *const ptr,讀為一個(gè)const指針,指向T類型。不過話說回來,我一直沒有用過,也還沒有想到使用的理由。
8.指向指針的指針
評(píng):對(duì)這一點(diǎn)我一直提不起興趣,討厭這么復(fù)雜的冬冬。不過在實(shí)際中漸漸明白。如 vector my_circle; vector::iterator p=my_circle.begin(); for(;p!=my_circle.end();p++) {(*p)->draw();}
9.新式轉(zhuǎn)型操作符
摘要:更丑陋,更難用,并且威力較小。
評(píng):看到了這一章,我心里面咯噔了一下。我的程序里面還有那么野蠻的、不講理的轉(zhuǎn)換。我得盡快用這種更丑陋的方式去改過來,以減少錯(cuò)誤的可能性。要注意static_const和dynamic_cast;
10.常量成員函數(shù)含義
摘要:
class X{
Public: void modify_buffer(int index,int val) const //不德!!!!!! { Buffer[index] = val;} Private: int *buffer; };
評(píng):的確是不道德的,const的函數(shù)偷偷的改了一個(gè)數(shù)據(jù),沒有發(fā)現(xiàn)因?yàn)闆]有修改class X中的對(duì)象。我得小心了,不干這種事情。
11.編譯器會(huì)在類中放東西
摘要:如果一個(gè)類聲明了一個(gè)或多個(gè)虛函數(shù),那么編譯器將會(huì)為該類的每一個(gè)對(duì)象插入一個(gè)指向虛函數(shù)表的指針。
摘要:如果使用了虛擬繼承,對(duì)象將會(huì)通過嵌入的指針、嵌入的偏移或其他的信息來保持對(duì)其虛基類子對(duì)象位置的跟蹤。
摘要:一個(gè)POD(“plain old data”)非常重要。比如int、double、C struct、union都是POD。摘要:如果希望復(fù)制一個(gè)類對(duì)象,那么永遠(yuǎn)都不要使用memcpy這樣的標(biāo)準(zhǔn)內(nèi)存塊復(fù)制函數(shù)。......相反,應(yīng)該使用對(duì)象的初始化或者賦值操作。
評(píng):看到最后這條我才明白為何有這章。看來對(duì)于高層的東西要使用高層的操作。
12.賦值和初始化并不相同
摘要:直截了當(dāng)?shù)恼f,賦值發(fā)生于當(dāng)你賦值時(shí),除此之外遇到的所有其他的復(fù)制情況均為初始化,包括聲明,函數(shù)返回,參數(shù)傳遞以及捕俘描述異常中的初始化。
評(píng):突然想起鼻祖書中的話,賦值是對(duì)一個(gè)結(jié)構(gòu)良好的存儲(chǔ)區(qū)去做,而初始化是對(duì)一個(gè)未定義的存儲(chǔ)區(qū)去做。如果要賦值,那么就確立一下賦值的對(duì)象是否還是磁盤上的荒蕪地帶。
13.復(fù)制操作
摘要:復(fù)制構(gòu)造和復(fù)制賦值是兩種不同的操作。
14.函數(shù)指針
摘要:將一個(gè)函數(shù)的地址初始化或賦值給一個(gè)指向函數(shù)的指針時(shí),無需顯示的取得函數(shù)地址。
例子:void (*fp)(int);
extern void h(int);
fp = h; //OK
fp = &h; //OK
摘要:為了調(diào)用函數(shù)指針?biāo)赶虻暮瘮?shù),而對(duì)指針進(jìn)行解引用操作也是不必要的。
例子:(*fp)(12); //顯示調(diào)用
fp(12); //隱式調(diào)用
摘要:函數(shù)指針的一個(gè)傳統(tǒng)用途是實(shí)現(xiàn)回調(diào)(callback)。
評(píng):一個(gè)很好的例子就是windows中對(duì)左右手的變化,如果改變了就只需要交換一下函數(shù)指針就OK了。
15.指向類成員的指針并非指針
摘要:“指向類成員的指針”即不包含地址,行為也不像指針。通常看作一個(gè)偏移量。
摘要:指向數(shù)據(jù)成員的指針對(duì)于描述“逆變性”的概念很方便。
摘要:存在指向基類成員的指針到指向公有派生類成員的指針隱式轉(zhuǎn)換,反之不行。
評(píng):明白了它的作用卻不知道哪里有用,完全可以換個(gè)方式使用,即使是為了上一章的回調(diào)。如果要得到類中的成員,完全可以用類的對(duì)象或指向類對(duì)象的指針獲得,這個(gè)可能是為了C程序員吧。
tip:不能想著指向static成員,它們并不是存在于所有類對(duì)象的一個(gè)偏移量。使用的方式和普通的數(shù)據(jù)一樣。
16.指向成員函數(shù)的指針并非指針
摘要:NULL
17.處理函數(shù)和數(shù)組聲明
摘要:
int *f1(); //一個(gè)返回值為int *的函數(shù)
int (*fp1)(); //一個(gè)指針,指向一個(gè)返回值為int的函數(shù)
int *a1[N]; //一個(gè)具有N個(gè)int *元素的數(shù)組
int (*ap1)[N]; //一個(gè)指針,指向一個(gè)具有N個(gè)int元素的數(shù)組
評(píng):夠復(fù)雜吧,不過看了下面的就要。。。。。。
摘要: int(*af2[N])(); //一個(gè)具有N個(gè)元素的數(shù)組
//其元素類型指向返回值為int的函數(shù)指針
評(píng):夠嚇人的,雖然有解決法子可我沒在意。因?yàn)槔显缇蜎Q定不用數(shù)組了。
18.函數(shù)對(duì)象
摘要:函數(shù)對(duì)象也是一個(gè)普通的類對(duì)象,通過重載函數(shù)調(diào)用操作符()來創(chuàng)建類似于函數(shù)指針的東西。
評(píng):這可是個(gè)好東西,在泛型算法中作為謂詞。
19.Command模式與好萊塢法則
摘要:好萊塢法則即“不要call我們,我們會(huì)call你”。
摘要:將一個(gè)函數(shù)對(duì)象于好萊塢法則相結(jié)合,即為Command模式的一個(gè)實(shí)例。
摘要:好處是,函數(shù)對(duì)象可以封裝數(shù)據(jù),另一個(gè)好處是函數(shù)對(duì)象可以通過虛擬成員表現(xiàn)出動(dòng)態(tài)行為,第三個(gè)好處是處理類層次結(jié)構(gòu)而不是較為原始的,缺乏靈活性的結(jié)構(gòu)(例如函數(shù)指針)。
20.STL函數(shù)參數(shù)
摘要:NULL
21.重載與重寫并不相同
摘要:重載發(fā)生于同一個(gè)作用域內(nèi)有兩個(gè)或更多個(gè)函數(shù)具有相同的名字但簽名不同時(shí)。
摘要:重寫發(fā)生于派生類函數(shù)和基類有相同的的名字和簽名時(shí)。
22.Template Method 模式
摘要: Template Method(模板方式)模式和C++模板一點(diǎn)關(guān)系都沒有。實(shí)際上,它是基類設(shè)計(jì)者為派生類設(shè)計(jì)者提供清晰指示的一種方式,這個(gè)指示就是“應(yīng)該如何實(shí)現(xiàn)基類所規(guī)定的契約”。
摘要:一個(gè)基類的成員函數(shù)是否應(yīng)該為非虛擬的、虛擬的或純虛擬的,這樣的決策主要是基于該函數(shù)的行為如何被派生類定制。
摘要:如果基類成員是非虛擬的,那么基類設(shè)計(jì)者就是以該基類為所確立的層次結(jié)構(gòu)指明了一個(gè)不變式。派生類不應(yīng)該用同名的派生類成員去隱藏基類非虛函數(shù)。
摘要:虛函數(shù)和純虛函數(shù)指定的操作,其實(shí)現(xiàn)可以由派生類通過重寫機(jī)制定制。一個(gè)非純虛函數(shù)提供了一個(gè)默認(rèn)實(shí)現(xiàn),并不強(qiáng)迫派生類一定要重寫它,而一個(gè)純虛函數(shù)則必須在具體派生類中進(jìn)行重寫。
評(píng):這一章對(duì)于我們?nèi)绾味x基類有了一個(gè)很好的說明。函數(shù)是具體的實(shí)現(xiàn),而我們確定如何具體的實(shí)現(xiàn)函數(shù)。
Tip:另外有一種派生類作為基類接口的形式,那里的法則就不太一樣了。基類是實(shí)現(xiàn),派生類是為了給其他用戶的接口。
23.名字空間
摘要:本質(zhì)上,名字空間是對(duì)全局作用域的細(xì)分。
摘要:許多C++程序員建議將using指令放在全局作用域中,這是個(gè)餿注意。
評(píng):一定要很好的區(qū)分什么是using指令和using聲明。
Using namespace namespace_name //是指令
Using anamespace_name::名字空間聲明的的東西 //是聲明
如果在全局作用域中使用using指令那等于去掉了名字空間的作用域。這里一般指的是程序員自己定義的名字空間,不是默認(rèn)的std。
24.成員函數(shù)查找
摘要:調(diào)用一個(gè)成員函數(shù)時(shí),涉及三個(gè)步驟:第一步,編譯器查找函數(shù)的名字;第二步,從可用候選者中選取最佳匹配函數(shù);第三步,檢查是否具有訪問該匹配函數(shù)是權(quán)限;
評(píng):這里面隱含的說了一個(gè)冬冬,如果編譯器找到了函數(shù)的名字,它是不會(huì)再去找的了。如果無權(quán)限范圍該函數(shù),那么就會(huì)在第三步出現(xiàn)編譯錯(cuò)誤。
25.實(shí)參相依的查找
摘要:ADL(實(shí)參相依的查找)指的是,當(dāng)查找一個(gè)函數(shù)調(diào)用表達(dá)式中的函數(shù)名字時(shí),編譯器也會(huì)到“包含函數(shù)調(diào)用實(shí)參的類型”的名字空間查找。
評(píng):我卻好像記得不止是名字空間,也包括class,因?yàn)閏lass其實(shí)也是一種特殊的名字空間。回頭還得再看一遍鼻祖的書,那里面其實(shí)都有,只是沒有重點(diǎn)標(biāo)出來而已。
26.操作符函數(shù)查找
摘要:當(dāng)使用函數(shù)調(diào)用語法時(shí),應(yīng)用的是普通的查找規(guī)則(ADL)。而對(duì)重載操作符的中綴調(diào)用的處理機(jī)制不同。
例子:
class X{ X operator%(const X&)const;};
X x,y;
x % y; //中綴調(diào)用
x.operator%(y); //成員函數(shù)調(diào)用
摘要:對(duì)于中綴操作符調(diào)用來說,編譯器不僅會(huì)考慮成員操作符,也會(huì)考慮非成員操作符。
評(píng):說的有點(diǎn)讓人迷糊。我想主要注意使用.operator時(shí),記得有個(gè)默認(rèn)的*this實(shí)參。
27.能力查詢
摘要:能力查詢只是偶爾需要,但它們往往被過渡使用。它們通常是糟糕設(shè)計(jì)的“指示器”。最好避免對(duì)一個(gè)對(duì)象的能力進(jìn)行運(yùn)行期查詢。
評(píng):能力查詢指的是對(duì)一個(gè)類對(duì)象進(jìn)行dynamic_cast,來知道它是否是另外的類型,通常,是橫向轉(zhuǎn)換而不是普通的向上或者向下。
28.指針比較的含義
摘要:指針比較不是關(guān)于地址的問題,而是關(guān)于對(duì)象同一性的問題。
摘要:一個(gè)非常重要的經(jīng)驗(yàn),處理指向?qū)ο蟮囊没蛑羔槙r(shí),必須小心避免丟失類型信息(如把指針賦值給void*指針)。
評(píng):一個(gè)基類的指針是與其派生類的指針==的,并不是因?yàn)榈刂废嗤穷愋拖嗤驗(yàn)榕缮惥褪腔悾拖癜嚅L(zhǎng)就是學(xué)生一樣。這里比較的是基類的類型。
29.虛構(gòu)造函數(shù)與Prototype模式
評(píng):構(gòu)造函數(shù)是不能虛的,而這里指的是具有這樣功能的函數(shù)。如在一個(gè)類中,我們使用一個(gè)成員函數(shù)clone來調(diào)用復(fù)制構(gòu)造函數(shù)new X(*this),我的經(jīng)驗(yàn)無法告訴我為何需要,不過有一點(diǎn)是很明確的,這個(gè)例子證明了軟件設(shè)計(jì)的“不知情”模式。
30.Factory Method模式
評(píng):一個(gè)沉重的打擊,我在28、29看見了什么是比較高級(jí)的構(gòu)架,也讓我想起了別來call我,需要時(shí)我來call你的好萊塢模式。每個(gè)類都明白自己做什么,而你只是在問它們一個(gè)很大眾的問題,而不是很私人的問題。重點(diǎn)推薦,我得好好看看,認(rèn)真感受。如果能很明確的使用,我的認(rèn)識(shí)將能夠上一個(gè)檔次。
31.協(xié)變返回類型
摘要:協(xié)變的優(yōu)勢(shì)在于,總是可以在適當(dāng)程度的抽象層工作。如果我們是處理Shape,獲得一個(gè)抽象的ShapeEditor;如果在處理某種具體的形狀類型,比如Circle,我們就可以直接獲得CircleEditor。協(xié)變機(jī)制使得我們可以不使用類型轉(zhuǎn)換操作來“重新”提供類型信息,而這種信息是一開始就不應(yīng)該丟掉的。
32.禁止復(fù)制
摘要:訪問修飾符(public、protect、private)可以用于表達(dá)和執(zhí)行高級(jí)約束技術(shù),指明一個(gè)類可以被怎樣使用。這些技術(shù)中最常見的一種是不接受對(duì)象的復(fù)制操作,這是通過將其復(fù)制操作聲明為private同時(shí)不為之提供定義而做到的。
評(píng):有時(shí)候我們應(yīng)該把所有不想給其他人使用的函數(shù)全部放進(jìn)private。
(因?yàn)楹竺娴谋容^深,暫時(shí)作罷。勉強(qiáng)看也只能懂個(gè)浮淺的東西。未完待續(xù))