探秘C++的復(fù)雜性問題
你用一個(gè)天生用來開啤酒瓶的工具開了啤酒瓶,不但啥成就感也沒有,而且誰也不會(huì)覺得你牛13。然而,如果你發(fā)明了一種用兩根筷子也能打開啤酒瓶的辦法,或者你干脆生就一口好牙可以把瓶蓋啃開,那也許就大不一樣了。人家就會(huì)覺得你很好很強(qiáng)大。
每個(gè)人都喜歡戴著腳鐐跳舞。
也就是說,如果你用一個(gè)天生為某個(gè)目的的工具來做他該做的事情,沒有人會(huì)喝彩,你也不會(huì)覺得了不起。但如果你用兩個(gè)本身不是為某個(gè)目的的工具組合出新功能的話,你就是“創(chuàng)新”者(盡管也許本來就有某個(gè)現(xiàn)成的工具可用)。
而C++則是這些“創(chuàng)新”的土壤,是的,我說的就是無窮無盡的workarounds和慣用法。但問題是,這些“創(chuàng)新”其實(shí)根本不是創(chuàng)新,你必須認(rèn)識(shí)到的是,他們都只不過是在沒有first-class解決方案的前提下不得已折騰出來的替補(bǔ)方案。是的,它們某種程度上的確可以叫創(chuàng)新,甚至研究可行的解決方案本身也是一件非常有意思的事情,但—
我知道它們很有趣,但實(shí)際上它們只是補(bǔ)丁方案。
是的,不要因?yàn)檫@些“創(chuàng)新”方案有趣就忍不住一頭鉆進(jìn)去。你之所以覺得有趣是因?yàn)楫?dāng)你一定程度上熟悉了C++之后,C++的所有一切,包括缺陷,對(duì)你來說就成了一個(gè)“既定事實(shí)”,一個(gè)背景,一個(gè)習(xí)以為常的東西(人是有很強(qiáng)的適應(yīng)性的)。因此,當(dāng)你發(fā)現(xiàn)在這個(gè)習(xí)以為常的環(huán)境下居然出現(xiàn)了新的可能性時(shí),你當(dāng)然是會(huì)歡呼雀躍的(比如我當(dāng)年讀《Modern C++ Design》的時(shí)候就有一次從早讀到晚,午飯都沒吃),然而實(shí)際上呢?其它語言中也許早就有first-class的支持了,其它語言也許根本不需要這個(gè)慣用法,因?yàn)樗鼈兙蜎]有這些缺陷。此外,從實(shí)踐的角度來說,更重要的是,這些“解決方案”也許你平時(shí)編程根本就用不到。
不,我當(dāng)然不是說這些補(bǔ)丁方案不重要。正如前面所說,C++中繁雜的技巧并非空穴來風(fēng),總有實(shí)際問題在背后驅(qū)動(dòng)的。但問題是,對(duì)于我們?nèi)粘>幊虂碚f,這些“實(shí)際問題”簡直是八桿子打不著的。犯不著先費(fèi)上80%的勁兒把20%時(shí)候才用到的東西揣在腦子里,用的時(shí)候查文檔或書就行了。 #t#
看到這里,塑造C++中特定的心態(tài)哲學(xué)的另一個(gè)原因想必你也已經(jīng)知道了。實(shí)際上,這個(gè)原因才是真正根本的。前面說的一個(gè)原因是C++書籍市場(教育)造就的,然而為什么人們喜歡寫這些書呢?進(jìn)一步說,為什么人們喜歡讀這些書呢?(我承認(rèn),我也曾經(jīng)讀得津津有味。)答案很簡單:心理。每個(gè)人都喜歡戴著腳鐐跳舞(事實(shí)8)。認(rèn)識(shí)到這一點(diǎn)不是為了提倡它,而是只有當(dāng)我們認(rèn)識(shí)到自己為什么會(huì)津津有味地去鉆研一堆補(bǔ)丁解決方案的時(shí)候,我們才真正能夠擺脫它們的吸引。
總而言之,C++的復(fù)雜性只是一個(gè)必要條件,并非問題的根本癥結(jié)。根本癥結(jié)在于人的心理,每個(gè)人都喜歡戴著腳鐐跳舞,并且以為是“創(chuàng)新”。意識(shí)到這一點(diǎn)之后可以幫我們避免被各種各樣名目繁多的語言細(xì)節(jié)和技巧占去不必要的時(shí)間。
然而,C++的復(fù)雜性始終是一個(gè)不可回避的現(xiàn)實(shí)。C++中有大量的陷阱和缺陷,后者導(dǎo)致了數(shù)目驚人的慣用法和workarounds。不加選擇的全盤預(yù)先學(xué)習(xí),是非常糟糕的做法,不僅低效,而且根本沒有必要,實(shí)在是浪費(fèi)生命。愛因斯坦曾經(jīng)說過,“我只想知道‘他’(宇宙)的設(shè)計(jì)理念,其它的都是細(xì)節(jié)”。然而,正如另一些讀者指出的,如果對(duì)C++中的這些細(xì)節(jié)事先一點(diǎn)都沒有概念的話,那么實(shí)際編碼中一旦遇到恐怕就變成沒頭蒼蠅了,也許到哪里去RTFM都不知道。這也是為什么那么多C++面試都會(huì)不厭其煩地問一些有代表性的語言細(xì)節(jié)的原因。
把細(xì)節(jié)全盤裝在腦子里固然不好,但對(duì)細(xì)節(jié)一無所知同樣也不是個(gè)辦法。那么對(duì)于C++程序員來說,在學(xué)習(xí)中究竟應(yīng)該以怎樣的態(tài)度和學(xué)習(xí)方法來對(duì)付C++的復(fù)雜性呢?其實(shí)答案也非常簡單,首先有一些很重要&必須的語言細(xì)節(jié)&特性是需要掌握的,然后我們只需知道在C++中大抵有哪些地方有復(fù)雜性(陷阱、缺陷),那么遇到問題的時(shí)候自然能夠知道到哪兒去尋找答案了。具體的建議在后文。