為什么說面向?qū)ο缶幊毯秃瘮?shù)式編程都有問題
我不理解為什么人們會對面向?qū)ο缶幊毯秃瘮?shù)式編程做無休無止的爭論。就好象這類問題已經(jīng)超越了人類智力極限,所以你可以幾個世紀的這樣討論下去。經(jīng) 過這些年對編程語言的研究,我已經(jīng)清楚的看到了問題的答案,所以,我經(jīng)常的發(fā)現(xiàn),人們對這些問題做的都是一些抓不住要領(lǐng)、無意義的爭論。
簡言之,不論是面向?qū)ο缶幊踢€是函數(shù)式編程,如果你走了極端,那都是錯誤的。面向?qū)ο缶幊痰臉O端是一切都是對象(純面向?qū)ο?。函數(shù)式編程的極端是純函數(shù)式編程語言。
面向?qū)ο缶幊痰膯栴}
面向?qū)ο蟮膯栴}在于它對“對象”的定義,它試圖將所有事情就納入到這個概念里。這種做法極端化后,你就得出來一個一切皆為對象思想。但這種思想是錯誤的,因為
有些東西不是對象。函數(shù)就不是對象。 |
也許你會反駁,在Python和Scala語言里,函數(shù)也是對象。在Python中,所有的含有一個叫做__call__的方法的對象其實都是函數(shù)。類似的,在Scala語言里,函數(shù)是擁有一個叫做apply方法的對象。但是,經(jīng)過認真的思考后,你會發(fā)現(xiàn),它混淆了源祖和衍生物的概念。函數(shù)是源祖,包含函數(shù)的對象實際是衍生物。__call__
和apply
它們自身首 先就是要定義的所謂“函數(shù)對象”。Python和Scala實際上是綁架了函數(shù),把它們監(jiān)禁在“對象”里,然后打上“__call__” 和 “apply” 標(biāo)簽,把它們稱作“方法”。當(dāng)然,如果你把一個函數(shù)封裝到對象里,你可以像使用一個函數(shù)那樣使用對象,但這并不意味著你可以說”函數(shù)也是對象“。
大 多數(shù)的面向?qū)ο笳Z言里都缺乏正確的實現(xiàn)一等(first-class)函數(shù)的機制。Java語言是一個極致,它完全不允許將函數(shù)當(dāng)作數(shù)據(jù)來傳遞。你可以將 全部的函數(shù)都封裝進對象,然后稱它們?yōu)?ldquo;方法”,但就像我說的,這是綁架。缺乏一等函數(shù)是為什么Java里需要這么多“設(shè)計模式”的主要原因。一旦有了一 等函數(shù),你將不再需要大部分的這些設(shè)計模式。
函數(shù)式編程的問題
相似的,函數(shù)式編程走向極端、成為一種純函數(shù)式編程語言后,也是有問題的。為了討論這個問題,我們最好先理解一下什么是純函數(shù)式編程語言。出于這個目的,你可能需要閱讀一下Amr Sabry先生(他是我的博士導(dǎo)師)的What is a Purely Functional Language。概述一下就是,純函數(shù)式編程語言是錯誤的,因為
有些東西不是純的。副作用是真實存在的。 |
所謂純函數(shù),基本上就是忽略了物質(zhì)基礎(chǔ)(硅片、晶體等)表現(xiàn)的特性。純函數(shù)式的編程語言試圖通過函數(shù)——在函數(shù)中傳入傳出整個宇宙——來重新實現(xiàn)整個宇宙。但物理的和模擬的是 有區(qū)別的。“副作用”是物理的。它們真實的存在于自然界中,對計算機的效用的實現(xiàn)起著不可或缺的作用。利用純函數(shù)來模擬它們是注定低效的、復(fù)雜的、甚至是 丑陋的。你是否發(fā)現(xiàn),在C語言里實現(xiàn)一個環(huán)形數(shù)據(jù)結(jié)構(gòu)或隨機數(shù)發(fā)生器是多么的簡單?但使用Haskell語言就不是這樣了。
還有,純函數(shù)編程語言會帶來巨大的認知成本。如果你深入觀察它們,你會看到monads
使程序變得復(fù)雜,難于編寫,而且monad的變體都是拙劣的修改。monads跟Java的“設(shè)計模式”具有相同的精神本質(zhì)。使用monad來 表現(xiàn)副作用就像是visitor模式來寫解釋器。你是否發(fā)現(xiàn),在很多其它語言里很簡單的事情,放到Haskell語言就變成了一個課題來研究如何實現(xiàn)?你 是否經(jīng)常會看到一些有著諸如“用Monadic的方式解決一個已經(jīng)解決的問題”這樣標(biāo)題的論文?有趣的是,Amr Sabry先生一起合著了這樣一篇論文。他試圖用Haskell語言重新實現(xiàn)Dan Friedman的miniKanren,但他不知道如何構(gòu)造這些monads
。 他向Oleg Kiselyov——公認的世界上對Haskell類型系統(tǒng)知識最淵博的人——求教。而且你可能不知道,Amr Sabry先生應(yīng)該是世界上對純函數(shù)編程語言知識最淵博的人了。他們在 Oleg 的幫助下解決了疑難后一起合著了這篇論文。諷刺的是,Dan Friedman——這個程序的原作者——在使用Scheme語言開發(fā)時卻沒有遇到任何問題。我在Dan的代碼基礎(chǔ)上重新實現(xiàn)了miniKanren,增 加了一個復(fù)雜的負操作。為了實現(xiàn)這個,我需要使用約束式邏輯編程和其它一些高級的技巧。鑒于用Haskell語言重寫基本的miniKanren將兩位世界級程序員都難倒了的事實,我不敢想象如果用Haskell的monads
如何能實現(xiàn)這些。
有些人認為monads
的價值在于,它們“圈定”了副作用的范圍。但如果monads
不能真正的使程序變得易于分析或更安全,這種“圈定”有什么用呢?事實上就是沒用處。本身就跟副作用一樣難于分析理解。沒有一種東西可以說monads
能使其簡單而靜態(tài)分析辦不到的。所有的靜態(tài)分析研究者都知道這點。靜態(tài)分析利用了monads
的本質(zhì),但卻去除了程序員編寫monads
代碼的負擔(dān)——而不是增加負擔(dān)。當(dāng)然,過度的副作用會使程序很難分析,但你也可以使用C語言寫出純函數(shù),例如:
- int f(int x) {
- int y = 0;
- int z = 0;
- y = 2 * x;
- z = y + 1;
- return z / 3;
- }
你用匯編語言也能做到這些。純函數(shù)并不專屬于純函數(shù)式編程語言。你可以用任何語言寫出純函數(shù),但重要的是,你必須也應(yīng)該允許副作用的存在。
回 首歷史,你會發(fā)現(xiàn),數(shù)學(xué)上的理想主義是純函數(shù)編程語言的背后推動力。數(shù)學(xué)函數(shù)簡單漂亮,但不幸的是,它們只是在你構(gòu)建原始純粹的模型時才好用。否者它們會 變得很丑陋。不要被“范疇論”等標(biāo)語嚇倒。我對范疇論了解很多。即使是范疇理論學(xué)家自己也稱其為“抽象無意義”,因為它們基本上就是用一種怪誕的方式告訴 你一些你已經(jīng)知道的事情!如果你讀過Gottlob Frege的文章Function and concept,你會吃驚的發(fā)現(xiàn),在他的這篇論文前的大多數(shù)數(shù)學(xué)家都錯誤的理解了函數(shù),而這僅僅是剛剛100多年前的事。事實上,數(shù)學(xué)語言上的很多事情都是有問題的。特別是微積分方面。編程語言的設(shè)計者們沒有理由要盲目的學(xué)習(xí)數(shù)學(xué)界。
不要盲目的愛上你的模型
無 論任何事情,當(dāng)走向極端時都是有害的。極端化時,面向?qū)ο缶幊毯秃瘮?shù)式編程都試圖把整個世界裝入它們的特有模型中,但這個世界是在完全不依賴我們的大腦思 考的情況下運轉(zhuǎn)的。如果以為你有一個錘子,就把所有東西都當(dāng)成釘子,這明顯是不對的。只有通過認清我們的真實世界,才能擺脫信仰對我們的束縛。
不要讓世界適應(yīng)你的模型。讓你的模型適應(yīng)世界。 |