掃二維碼與項目經(jīng)理溝通
我們在微信上24小時期待你的聲音
解答本文疑問/技術(shù)咨詢/運營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流
本文轉(zhuǎn)載自公眾號“讀芯術(shù)”(ID:AI_Discovery)。

創(chuàng)新互聯(lián)公司提供高防物理服務器租用、云服務器、香港服務器、成都聯(lián)通服務器托管等
當我們提及Python時,常常指的是CPython,即C語言實現(xiàn)的Python,這就是PyPy發(fā)揮作用的地方啦。實話講,Python很慢,而用Python(!)編寫的Python在運行時,執(zhí)行速度比CPython快4.4倍。這是如何實現(xiàn)的呢?
正如Python之父吉多·范羅蘇姆(Guido van Rossum)所說:“如果想讓代碼運行得更快,應該使用PyPy?!?/p>
第一批使用PyPy的程序員競爭力很強。有時候,就算花了很多時間用Python編寫解決方案,最后也會失敗,但是在PyPy中運行同樣的代碼就會成功。這是為什么?
不同的方法
Python是一種解釋性語言,CPython逐行讀取并執(zhí)行源代碼。解釋性語言(包括JavaScript)有很多優(yōu)點:
當然,它也有一些缺點:
可以看到,其優(yōu)缺點都包括了“編譯不會失敗”。這有時候基于不同的行為(比如原型設(shè)計或生產(chǎn)),但我仍然傾向于將其視為一個缺點。而PyPy稍有不同,它不是一個純粹的解釋器,而是實現(xiàn)了跟蹤即時(JIT)編譯。
即時編譯
即時編譯介于解釋和傳統(tǒng)提前編譯之間。即時編譯器并不執(zhí)行源代碼本身,而是生成一組可立即執(zhí)行的低層指令(通常是匯編)。
這個插圖可以幫助我們理解其中的區(qū)別。在編譯語言(C、C++、Rust)中,編譯階段嚴格按照開發(fā)環(huán)境劃分。它生成一個可運行的二進制文件,然后將其發(fā)送到生產(chǎn)環(huán)境中。
在解釋性語言中,情況正好相反:源代碼(在*化之后,hello JS)被全部推到生產(chǎn)環(huán)境中,由解釋器執(zhí)行。即時編譯語言也運送源代碼(或字節(jié)碼,如Java或C#),但它是作為一種常規(guī)編譯語言編譯和運行,而不是逐行解釋。
并不是說一種方法比另一種好,每個用例都將根據(jù)其獨特的需求指定正確的選擇。但是,如果性能非常關(guān)鍵,使用Python解釋器感覺很舒服,那么你可以選擇PyPy。
跟蹤即時編譯
與編譯或解釋一樣,實現(xiàn)即時編譯也有不同的方法。傳統(tǒng)的方法是方法/函數(shù)作用域。當代碼調(diào)用一個函數(shù)時,即時編譯器將獲得它的源代碼,進行編譯并提供可執(zhí)行的二進制文件。而PyPy采用的方法稍微不同,這是由Python的特性和用例決定的。
PyPy的編譯器不是按方法調(diào)用,而是計算循環(huán)。由于Python在數(shù)據(jù)科學、機器學習、高級算法以及數(shù)據(jù)結(jié)構(gòu)中使用廣泛,這是最有意義的。簡而言之,PyPy是Python之上的一個優(yōu)化層。
PyPy并不像所理解的那樣嚴格處理循環(huán)。除了常規(guī)的for和while結(jié)構(gòu)外,如果PyPy檢測到編譯工作有價值,它會優(yōu)化任意代碼塊。
缺點
圖源:unsplash
當然,PyPy也有缺點。即使它能大大提高性能,你也需要知道以下幾點:
與任何工具一樣,在采用之前,我們應該考慮所有的細節(jié)。不過,下次登錄到Codeforces接受挑戰(zhàn)時,不妨試用一下PyPy。有可能O(n^3)錯誤代碼會通過,而在純Python中,只有O(n log n)會通過。
源代碼
雖然本文沒有涉及CPython和PyPy的源代碼,但是這些文件在CPython (C代碼)和PyPy (Python代碼)中實現(xiàn)了階乘函數(shù)。
其他實現(xiàn)
除了CPython和PyPy,還有其他值得注意的Python實現(xiàn):
去試試使用PyPy來體驗代碼加速度吧!

我們在微信上24小時期待你的聲音
解答本文疑問/技術(shù)咨詢/運營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流