python是我喜歡的語言,簡(jiǎn)潔,優(yōu)美,容易使用。前兩天,我很激昂的向朋友宣傳python的好處。
聽過之后,朋友問我:好吧,我承認(rèn)python不錯(cuò),但它為什么叫python呢?
我不是很確定:呃,似乎是一個(gè)電視劇的名字。
朋友又問:那你說的guido是美國(guó)人么? (guido von rossum,python的作者)
我再次不是很確定:他從google換到dropbox工作,但他的名字像是荷蘭人的 (有一個(gè)von在中間)。
所以,后面我花了些時(shí)間調(diào)查python的歷史。這是很好的學(xué)習(xí)。我看到了python中許多功能的來源和python的設(shè)計(jì)理念,比如哪些功能是歷史遺留,哪些功能是重復(fù),如何增加功能…… 而且,python也是開源(open source)運(yùn)動(dòng)的一個(gè)成功案例。從python的歷史中,我們可以一窺開源開發(fā)的理念和成就。
python的起源
python的作者,guido von rossum,確實(shí)是荷蘭人。1982年,guido從阿姆斯特丹大學(xué)(university of amsterdam)獲得了數(shù)學(xué)和計(jì)算機(jī)碩士學(xué)位。然而,盡管他算得上是一位數(shù)學(xué)家,但他更加享受計(jì)算機(jī)帶來的樂趣。用他的話說,盡管擁有數(shù)學(xué)和計(jì)算機(jī)雙料資質(zhì),他總趨向于做計(jì)算機(jī)相關(guān)的工作,并熱衷于做任何和編程相關(guān)的活兒。
guido von rossum
在那個(gè)時(shí)候,他接觸并使用過諸如pascal、c、 fortran等語言。這些語言的基本設(shè)計(jì)原則是讓機(jī)器能更快運(yùn)行。在80年代,雖然ibm和蘋果已經(jīng)掀起了個(gè)人電腦浪潮,但這些個(gè)人電腦的配置很低 (在今天看來)。比如早期的macintosh,只有8mhz的cpu主頻和128kb的ram,一個(gè)大的數(shù)組就能占滿內(nèi)存。所有的編譯器的核心是做優(yōu)化,以便讓程序能夠運(yùn)行。為了增進(jìn)效率,語言也迫使程序員像計(jì)算機(jī)一樣思考,以便能寫出更符合機(jī)器口味的程序。在那個(gè)時(shí)代,程序員恨不得用手榨取計(jì)算機(jī)每一寸的能力。有人甚至認(rèn)為c語言的指針是在浪費(fèi)內(nèi)存。至于動(dòng)態(tài)類型,內(nèi)存自動(dòng)管理,面向?qū)ο?hellip;… 別想了,那會(huì)讓你的電腦陷入癱瘓。
然而,這種思考方式讓guido感到苦惱。guido知道如何用c語言寫出一個(gè)功能,但整個(gè)編寫過程需要耗費(fèi)大量的時(shí)間 (即使他已經(jīng)準(zhǔn)確的知道了如何實(shí)現(xiàn))。他的另一個(gè)選擇是shell。bourne shell作為unix系統(tǒng)的解釋器(interpreter)已經(jīng)長(zhǎng)期存在。unix的管理員們常常用shell去寫一些簡(jiǎn)單的腳本,以進(jìn)行一些系統(tǒng)維護(hù)的工作,比如定期備份、文件系統(tǒng)管理等等。shell可以像膠水一樣,將unix下的許多功能連接在一起。許多c語言下上百行的程序,在shell下只用幾行就可以完成。然而,shell的本質(zhì)是調(diào)用命令。它并不是一個(gè)真正的語言。比如說,shell沒有數(shù)值型的數(shù)據(jù)類型,加法運(yùn)算都很復(fù)雜。總之,shell不能全面的調(diào)動(dòng)計(jì)算機(jī)的功能。
guido希望有一種語言,這種語言能夠像c語言那樣,能夠全面調(diào)用計(jì)算機(jī)的功能接口,又可以像shell那樣,可以輕松的編程。abc語言讓guido看到希望。abc是由荷蘭的cwi (centrum wiskunde & informatica, 數(shù)學(xué)和計(jì)算機(jī)研究所)開發(fā)的。guido在cwi工作,并參與到abc語言的開發(fā)。abc語言以教學(xué)為目的。與當(dāng)時(shí)的大部分語言不同,abc語言的目標(biāo)是“讓用戶感覺更好”。abc語言希望讓語言變得容易閱讀,容易使用,容易記憶,容易學(xué)習(xí),并以此來激發(fā)人們學(xué)習(xí)編程的興趣。比如下面是一段來自wikipedia的abc程序,這個(gè)程序用于統(tǒng)計(jì)文本中出現(xiàn)的詞(word)的總數(shù):
1
2
3
4
5
6
7
|
how to return words document: put {} in collection for line in document: for word in split line: if word not . in collection: insert word in collection return collection |
how to用于定義一個(gè)函數(shù)。一個(gè)python程序員應(yīng)該很容易理解這段程序。abc語言使用冒號(hào)(:)和縮進(jìn)來表示程序塊(c語言使用{}來表示程序塊)。行尾沒有分號(hào)。for和if結(jié)構(gòu)中也沒有括號(hào)()。如果將how to改為def,將put行改為collection = [],將insert行改為collection.append(word),這就幾乎是一個(gè)標(biāo)準(zhǔn)的python函數(shù)。上面的函數(shù)讀起來就像一段自然的文字。
盡管已經(jīng)具備了良好的可讀性和易用性,abc語言最終沒有流行起來。在當(dāng)時(shí),abc語言編譯器需要比較高配置的電腦才能運(yùn)行。而這些電腦的使用者通常精通計(jì)算機(jī),他們更多考慮程序的效率,而非它的學(xué)習(xí)難度。除了硬件上的困難外,abc語言的設(shè)計(jì)也存在一些致命的問題:
可拓展性差。abc語言不是模塊化語言。如果想在abc語言中增加功能,比如對(duì)圖形化的支持,就必須改動(dòng)很多地方。不能直接進(jìn)行io。abc語言不能直接操作文件系統(tǒng)。盡管你可以通過諸如文本流的方式導(dǎo)入數(shù)據(jù),但abc無法直接讀寫文件。輸入輸出的困難對(duì)于計(jì)算機(jī)語言來說是致命的。你能想像一個(gè)打不開車門的跑車么?過度革新。abc用自然語言的方式來表達(dá)程序的意義,比如上面程序中的how to (如何)。然而對(duì)于程序員來說,他們更習(xí)慣用function或者define來定義一個(gè)函數(shù)。同樣,程序員也習(xí)慣了用等號(hào)(=)來分配變量。這盡管讓abc語言顯得特別,但實(shí)際上增加了程序員的學(xué)習(xí)難度 (程序員大都掌握不止一種語言)。傳播困難。abc編譯器很大,必須被保存在磁帶(tape)上。當(dāng)時(shí)guido在訪問的時(shí)候,就必須有一個(gè)大磁帶來給別人安裝abc編譯器。 這樣,abc語言就很難快速傳播。
ibm tape drive:讀寫磁帶
1989年,為了打發(fā)圣誕節(jié)假期,guido開始寫python語言的編譯/解釋器。python來自guido所摯愛的電視劇monty python's flying circus (bbc1960-1970年代播放的室內(nèi)情景幽默劇,以當(dāng)時(shí)的英國(guó)生活為素材)。他希望這個(gè)新的叫做python的語言,能實(shí)現(xiàn)他的理念(一種c和shell之間,功能全面,易學(xué)易用,可拓展的語言)。guido作為一個(gè)語言設(shè)計(jì)愛好者,已經(jīng)有過設(shè)計(jì)語言的(不很成功)的嘗試。這一次,也不過是一次純粹的hacking行為。
python的誕生
1991年,第一個(gè)python編譯器(同時(shí)也是解釋器)誕生。它是用c語言實(shí)現(xiàn)的,并能夠調(diào)用c庫(.so文件)。從一出生,python已經(jīng)具有了:類(class),函數(shù)(function),異常處理(exception),包括表(list)和詞典(dictionary)在內(nèi)的核心數(shù)據(jù)類型,以及模塊(module)為基礎(chǔ)的拓展系統(tǒng)。
最初的python logo: 由guido的兄弟just von rossum設(shè)計(jì)
python語法很多來自c,但又受到abc語言的強(qiáng)烈影響。來自abc語言的一些規(guī)定直到今天還富有爭(zhēng)議,比如強(qiáng)制縮進(jìn)。但這些語法規(guī)定讓python容易讀。另一方面,python聰明的選擇服從一些慣例(特別是c語言的慣例)。比如使用等號(hào)賦值,使用def來定義函數(shù)。guido認(rèn)為,如果“常識(shí)”上確立的東西,沒有必要過度糾結(jié)。
python從一開始就特別在意可拓展性(extensibility)。python可以在多個(gè)層次上拓展。從高層上,你可以引入.py文件。在底層,你可以引用c語言的庫。python程序員可以快速的使用python寫.py文件作為拓展模塊。但當(dāng)性能是考慮的重要因素時(shí),python程序員可以深入底層,寫c程序,編譯為.so文件引入到python中使用。python就好像是使用鋼構(gòu)建房一樣,先規(guī)定好大的框架。而程序員可以在此框架下相當(dāng)自由的拓展或更改。
最初的python完全由guido本人開發(fā)。python得到guido同事的歡迎。他們迅速的反饋使用意見,并參與到python的改進(jìn)。guido和一些同事構(gòu)成python的核心團(tuán)隊(duì)。他們將自己大部分的業(yè)余時(shí)間用于hack python (也包括工作時(shí)間,因?yàn)樗麄儗ython用于工作)。隨后,python拓展到cwi之外。python將許多機(jī)器層面上的細(xì)節(jié)隱藏,交給編譯器處理,并凸顯出邏輯層面的編程思考。python程序員可以花更多的時(shí)間用于思考程序的邏輯,而不是具體的實(shí)現(xiàn)細(xì)節(jié) (guido有一件t恤,寫著:人生苦短,我用python)。這一特征吸引了廣大的程序員。python開始流行。
我們不得不暫停我們的python時(shí)間,轉(zhuǎn)而看一看這時(shí)的計(jì)算機(jī)概況。1990年代初,個(gè)人計(jì)算機(jī)開始進(jìn)入普通家庭。intel發(fā)布了486處理器,windows發(fā)布window 3.0開始的一系列視窗系統(tǒng)。計(jì)算機(jī)的性能大大提高。程序員開始關(guān)注計(jì)算機(jī)的易用性 (比如圖形化界面)。
windows 3.0
由于計(jì)算機(jī)性能的提高,軟件的世界也開始隨之改變。硬件足以滿足許多個(gè)人電腦的需要。硬件廠商甚至渴望高需求軟件的出現(xiàn),以帶動(dòng)硬件的更新?lián)Q代。c++和java相繼流行。c++和java提供了面向?qū)ο蟮木幊谭妒剑约柏S富的對(duì)象庫。在犧牲了一定的性能的代價(jià)下,c++和java大大提高了程序的產(chǎn)量。語言的易用性被提到一個(gè)新的高度。我們還記得,abc失敗的一個(gè)重要原因是硬件的性能限制。從這方面說,python要比abc幸運(yùn)許多。
另一個(gè)悄然發(fā)生的改變是internet。1990年代還是個(gè)人電腦的時(shí)代,windows和intel挾pc以令天下,盛極一時(shí)。盡管internet為主體的信息革命尚未到來,但許多程序員以及資深計(jì)算機(jī)用戶已經(jīng)在頻繁使用internet進(jìn)行交流 (包括email和newsgroup)。internet讓信息交流成本大大下降。一種新的軟件開發(fā)模式開始流行:開源 (open source)。程序員利用業(yè)余時(shí)間進(jìn)行軟件開發(fā),并開放源代碼。1991年,linus在comp.os.minix新聞組上發(fā)布了linux內(nèi)核源代碼,吸引大批hacker的加入。linux和gnu相互合作,最終構(gòu)成了一個(gè)充滿活力的開源平臺(tái)。
硬件性能不是瓶頸,python又容易使用,所以許多人開始轉(zhuǎn)向python。guido維護(hù)了一個(gè)maillist,python用戶就通過郵件進(jìn)行交流。python用戶來自許多領(lǐng)域,有不同的背景,對(duì)python也有不同的需求。python相當(dāng)?shù)拈_放,又容易拓展,所以當(dāng)用戶不滿足于現(xiàn)有功能,很容易對(duì)python進(jìn)行拓展或改造。隨后,這些用戶將改動(dòng)發(fā)給guido,并由guido決定是否將新的特征加入到python或者標(biāo)準(zhǔn)庫中。如果代碼能被納入python自身或者標(biāo)準(zhǔn)庫,這將極大的榮譽(yù)。python自身也因此變得更好。
(guido不得不作出許多決定,這也是他被稱為benevolent dictator for life的原因)
python被稱為“battery included”,是說它以及其標(biāo)準(zhǔn)庫的功能強(qiáng)大。這些是整個(gè)社區(qū)的貢獻(xiàn)。python的開發(fā)者來自不同領(lǐng)域,他們將不同領(lǐng)域的優(yōu)點(diǎn)帶給python。比如python標(biāo)準(zhǔn)庫中的正則表達(dá)(regular expression)是參考perl,而lambda, map, filter, reduce函數(shù)參考lisp。python本身的一些功能以及大部分的標(biāo)準(zhǔn)庫來自于社區(qū)。python的社區(qū)不斷擴(kuò)大,進(jìn)而擁有了自己的newsgroup,網(wǎng)站(python.org),以及基金 (python software foundation)。從python 2.0開始,python也從maillist的開發(fā)方式,轉(zhuǎn)為完全開源的開發(fā)方式。社區(qū)氣氛已經(jīng)形成,工作被整個(gè)社區(qū)分擔(dān),python也獲得了更加高速的發(fā)展。
(由于guido享有絕對(duì)的仲裁權(quán),所以在python早期maillist的開發(fā)時(shí)代,不少愛好者相當(dāng)擔(dān)心guido的生命。他們甚至作出假設(shè):如果guido掛了的話,python會(huì)怎樣)
到今天,python的框架已經(jīng)確立。python語言以對(duì)象為核心組織代碼(everything is object),支持多種編程范式(multi-paradigm),采用動(dòng)態(tài)類型(dynamic typing),自動(dòng)進(jìn)行內(nèi)存回收(garbage collection)。python支持解釋運(yùn)行(interpret),并能調(diào)用c庫進(jìn)行拓展。python有強(qiáng)大的標(biāo)準(zhǔn)庫 (battery included)。由于標(biāo)準(zhǔn)庫的體系已經(jīng)穩(wěn)定,所以python的生態(tài)系統(tǒng)開始拓展到第三方包。這些包,如django, web.py, wxpython, numpy, matplotlib,pil,將python升級(jí)成了物種豐富的熱帶雨林。
今天python已經(jīng)進(jìn)入到3.0的時(shí)代。由于python 3.0向后不兼容,所以從2.0到3.0的過渡并不容易。另一方面,python的性能依然值得改進(jìn),python的運(yùn)算性能低于c++和java。python依然是一個(gè)在發(fā)展中的語言。我期待看到python的未來。
python啟示錄
python崇尚優(yōu)美、清晰、簡(jiǎn)單,是一個(gè)優(yōu)秀并廣泛使用的語言 (tiobe語言排行第八,google的第三大開發(fā)語言,dropbox的基礎(chǔ)語言,豆瓣的服務(wù)器語言)。這個(gè)世界并不缺乏優(yōu)秀的語言,但python的發(fā)展史作為一個(gè)代表,帶給我許多啟示。
在python的開發(fā)過程中,社區(qū)起到了重要的作用。guido自認(rèn)為自己不是全能型的程序員,所以他只負(fù)責(zé)制訂框架。如果問題太復(fù)雜,他會(huì)選擇繞過去,也就是cut the corner。這些問題最終由社區(qū)中的其他人解決。社區(qū)中的人才是異常豐富的,就連創(chuàng)建網(wǎng)站,籌集基金這樣與開發(fā)稍遠(yuǎn)的事情,也有人樂意于處理。如今的項(xiàng)目開發(fā)越來越復(fù)雜,越來越龐大,合作以及開放的心態(tài)成為項(xiàng)目最終成功的關(guān)鍵。
python從其他語言中學(xué)到了很多,無論是已經(jīng)進(jìn)入歷史的abc,還是依然在使用的c和perl,以及許多沒有列出的其他語言。可以說,python的成功代表了它所有借鑒的語言的成功。同樣,ruby借鑒了python,它的成功也代表了python某些方面的成功。每個(gè)語言都是混合體,都有它優(yōu)秀的地方,但也有各種各樣的缺陷。同時(shí),一個(gè)語言“好與不好”的評(píng)判,往往受制于平臺(tái)、硬件、時(shí)代等等外部原因。程序員經(jīng)歷過許多語言之爭(zhēng)。我想,為什么不以開放的心態(tài)和客觀的分析,去區(qū)分一下每個(gè)語言的具體優(yōu)點(diǎn)缺點(diǎn),去區(qū)分內(nèi)部和外部的因素。說不定哪一天發(fā)現(xiàn),我不喜歡的某個(gè)語言中,正包含了我所需要的東西。
無論python未來的命運(yùn)如何,python的歷史已經(jīng)是本很有趣的小說。