主頁 > 前端設計 > 用 PyPy 讓你的 Python 代碼運行得更快!

用 PyPy 讓你的 Python 代碼運行得更快!

2020-11-15 17:24:01 前端設計

Python是開發人員中最常用的編程語言之一,但它有一定的局限性,例如,對于某些應用程式而言,它的運行速度可能比其它語言低100倍,這就是為什么當Python的運行速度成為用戶瓶頸后,許多公司會用另一種語言重寫他們的應用程式,但是有沒有一種方法既可以保持Python的特性又能提高速度呢?它就是PyPy

PyPy是一種非常兼容的Python解釋器,它是CPython2.7、3.6和即將推出的3.7的一種值得替代的方法,在安裝和運行應用程式時使用它,可以顯著提高速度,速度提高多少取決于你運行的應用程式,

在本教程中,您將學習:

  • 如何使用PyPy安裝和運行代碼

  • PyPy與CPython在速度方面的比較

  • PyPy的功能及其如何使Python代碼更快地運行

  • 本教程中的示例使用 Python 3.6 ,因為它是PyPy兼容的最新 Python 版本,

PyPy 簡介

Python解釋器可以用多種語言來實作,如CPython(用C撰寫)、Jython(用Java撰寫)、Iron Python(用.NET撰寫)和PyPy(用Python撰寫),

CPython是Python解釋器的最初實作,也是迄今為止使用最廣和最多維護的,當我們從Python官方網站下載并安裝好Python 3.x后,我們就直接獲得了一個官方版本的解釋器:CPython,這個解釋器是用C語言開發的,所以叫CPython,在命令列下運行python就是啟動CPython解釋器,

但是,由于CPython是一種高級的解釋語言,因此它有一定的局限性,并且在速度方面沒有任何優勢,這就是PyPy可以起作用的地方,由于它符合Python語言規范,因此Py Py不需要對代碼庫進行任何更改,并且可以通過下面的功能顯著提高速度,

現在,您可能想知道,如果CPython使用相同的語法,為什么它不實作Py Py的強大功能,原因是,實施這些功能需要對源代碼進行巨大的更改,這將是一項非常繁瑣的作業,

我們來粗略看一下如何在實際操作中使用PyPy,

安裝

您的作業系統可能已提供PyPy軟體包,例如,在Mac OS上,您可以在Homebrew的幫助下安裝它:

$ brew install pypy3

或者您也可以下載與作業系統匹配的二進制檔案,完成下載后,只需打開tarball或ZIP檔案即可,然后,您可以執行以下操作:

$ tar xf pypy3.6-v7.3.1-osx64.tar.bz2
$ ./pypy3.6-v7.3.1-osx64/bin/pypy3
Python 3.6.9 (?, Jul 19 2020, 21:37:06)
[PyPy 7.3.1 with GCC 4.2.1]
Type "help", "copyright", "credits" or "license" for more information.

您需要在上述檔案夾地址執行該命令,有關完整的說明,請參閱安裝檔案,

運行 PyPy

您現在已經安裝了Py Py,并且即將運行它!為此,請創建一個名為script.py的Python檔案,并將以下代碼放入其中:

total = 0
for i in range(1, 10000):
    for j in range(1, 10000):
        total += i + j

print(f"The result is {total}")

在兩個嵌套的for回圈中,將1到9,999之間的數字相加,并列印結果,

查看運行此腳本需要多長時間:

import time

start_time = time.time()

total = 0
for i in range(1, 10000):
    for j in range(1, 10000):
        total += i + j

print(f"The result is {total}")

end_time = time.time()
print(f"It took {end_time-start_time:.2f} seconds to compute")

該代碼現在執行以下操作:

  • 第3行將當前時間保存到變數start_time

  • 第5至8行運行回圈,

  • 第10行列印結果,

  • 第12行將當前時間保存為end_time

  • 第13行列印開始時間和結束時間之間的差值,以顯示運行腳本所需的時間,

用Python來運行它,下面是我在Mac Book Pro上的結果:

$ python3.6 script.py
The result is 999800010000
It took 20.66 seconds to compute

現在使用Py Py運行它:

$ pypy3 script.py
The result is 999800010000
It took 0.22 seconds to compute

在這個小實驗中,PyPy的速度大約是Python的94倍!

您可以通過瀏覽 PyPy Speed Center 來查看更多嚴格的測驗,

請記住,PyPy如何影響代碼的性能取決于您用代碼來做什么,在某些情況下,Py Py實際上較慢,稍后會看到,但是,就幾何平均而言,它的速度是Python的4.3倍,

PyPy及其特性

Py Py有兩種定義:

1、用于生成動態語言解釋器的動態語言框架 2、使用該框架的Python實作

您應該已經意識到了第二個問題,您使用的Python實作是使用稱為RPython的動態語言框架撰寫的,就像CPython是用C撰寫的,而Jython是用Java撰寫的一樣,

但之前文中不是提到PyPy是用Python撰寫的嗎?嗯,這有點簡單,PyPy成為用Python撰寫的Python解釋器(而不是RPython)這么說的原因是RPython使用了與Python相同的語法,

PyPy是怎么來的?需要解釋以下幾點:

1、它的源代碼是用RPython撰寫,

2、RPython轉換工具應用到了代碼中,從根本上提高了代碼效率,還可以將代碼編譯為機器代碼,這就是Mac,Windows和Linux用戶必須下載不同版本的原因,

3、用上述方式生成的二進制可執行檔案,就是你運行的Python解釋器,

你不需要執行上述所有這些步驟來使用PyPy,因為已經有提供您安裝和使用的可執行檔案,

此外,由于在框架和實作中使用同一個詞非常令人困惑,PyPy背后的團隊決定放棄這種雙重用法,現在,PyPy僅指Python解釋器,而框架被稱為RPython轉換工具,

接下來,您將了解在什么情況下使用PyPy比Python更好、更快,

Just-In-Time (JIT) 編譯器

在了解JIT編譯器的內容之前,讓我們先回顧一下已編譯語言(如C)和解釋語言(如JavaScript)的特性,

在編譯型語言寫的程式執行之前,需要一個專門的編譯程序,把源代碼編譯成機器語言的檔案,如exe格式的檔案,以后要再運行時,直接使用編譯結果即可,如直接運行exe檔案,因為只需編譯一次,以后運行時不需要編譯,所以編譯型語言執行效率高,與特定平臺相關,一般無法移植到其他平臺,如C、C++、Objective等都屬于編譯型語言,

解釋型語言不需要事先編譯,其直接將源代碼解釋成機器碼并立即執行,所以只要某一平臺提供了相應的解釋器即可運行該程式,解釋型語言每次運行都需要將源代碼解釋稱機器碼并執行,效率較低;只要平臺提供相應的解釋器,就可以運行源代碼,所以可以方便源程式移植,

然后還有一些編程語言,例如Python,它混合了編譯和解釋,具體來說,Python首先編譯為位元組碼,然后由CPython解釋,這使代碼的性能優于用純解釋型語言撰寫的代碼,并保持可移植性優勢,

但是它的性能仍然遠遠低于編譯型語言,其原因是,編譯后的代碼可以執行許多優化,而位元組碼是不可能的,

這就是JIT編譯器的來源,它試圖通過對機器代碼進行一些編譯和一些解釋來同時獲得兩種優勢,簡而言之,以下是JIT編譯為提供更快性能所采取的步驟:

1、識別代碼中最常用的組件,如回圈中的函式,

2、運行時將這些部件轉換為機器代碼,

3、優化生成的機器代碼,

4、用優化的機器代碼版本取代之前的實作,

還記得教程開頭的兩個嵌套回圈嗎?PyPy檢測到重復執行相同操作時,將其編譯為機器代碼,優化機器代碼,然后轉換實作,這也是為什么您會看到這樣的結果,

垃圾回識訓制

無論何時創建變數、函式或任何其他物件,您的計算機都會給它們分配記憶體,最終,其中一些物件將不再需要,如果不及時清理,計算機可能會耗盡記憶體并使程式崩潰,

在C和C++等編程語言中,通常必須手動處理此問題,其他編程語言(如Python和Java)會自動為您執行此操作,這被稱為自動垃圾回識訓制,

CPython使用一種稱為參考計數的技術,實質上,每當參考物件時,Python物件的參考計數都會增加,而在取消參考該物件時則遞減計數,當參考計數為零時,CPython會自動為該物件呼叫記憶體釋放函式,這是一種簡單有效的技術,但有一個陷阱,

當大型物件樹的參考計數變為零時,所有相關物件將被釋放,因此,您可能有很長的暫停時間,在此期間您的程式根本無法執行,

此外,還有一個例子,其中參考計數根本不起作用,如下所示:

class A(object):
    pass

a = A()
a.some_property = a
del a

在上面的代碼中,定義了新的類,然后,創建一個實體,并將其指定為其自身的屬性,最后,洗掉實體,

此時,實體將不再可訪問,但是,參考計數不會從記憶體中洗掉實體,因為它具有對自身的參考,因此參考計數不是零,此問題被稱為參考回圈,無法使用參考計數解決,

這是CPython使用的另一個工具,稱為回圈垃圾回收器,它從已知根(如型別物件)開始遍歷記憶體中的所有物件,然后,它標識所有可訪問的物件,并釋放不可訪問的物件,因為它們不再存在,這樣就解決了參考回圈問題,但是,當記憶體中存在大量物件時,它可能會創建更明顯的暫停,

另一方面,PyPy不使用參考計數,相反,它只使用第二種技術,即回圈查找器,也就是說,它會定期從根開始遍歷活動物件,這使PyPy比CPython具有一些優勢,因為它不需要考慮參考計數,從而使記憶體管理花費的總時間少于CPython,

此外,PyPy將作業拆分為可變數量的部分,并運行每個部分,直到沒有剩余部分為止,此方法只在每個次要集合之后添加幾毫秒,而不像CPython那樣一次添加數百毫秒,

垃圾回識訓制非常復雜,并且有許多超出本教程范圍的內容,您可以在檔案中找到有關PyPy垃圾回識訓制的詳細資訊,

PyPy的局限性

PyPy并非萬能,它不是一個適合您所有任務的工具,它甚至可能使應用程式的執行速度比CPython慢得多,這就是為什么您必須記住以下局限性,

它不適用于C擴展

PyPy最適合純Python應用程式,無論何時使用C擴展模塊,它的運行速度都要比在CPython中慢得多,原因是PyPy無法優化C擴展模塊,因為它們不受完全支持,此外,PyPy必須模擬代碼中的參考計數,使其更慢,

在這種情況下,PyPy團隊建議去掉CPython擴展并將其替換為純Python版本,如果不行的話,則必須使用CPython,

盡管如此,核心團隊正在處理C擴展,有些軟體包已被移植到PyPy,并且作業速度也同樣快,

它只適用于長時間運行的程式

想象一下你想去一家離你家很近的商店,您既可以直接走路前往,也可以開車,

您的車明顯比您的腳快得多,但是,請考慮需要您完成的步驟:

1.去你的車庫,

2、開車,

3、給車預熱,

4、開車去商店,

5、尋找停車位,

6、在回傳途中重復此程序,

開車需要一系列麻煩的步驟,如果你想去的地方就在附近,那就不一定值得了,

現在想想,如果你想去50公里外的鄰近城市,會發生什么?開車去那里肯定是值得的,而不是步行去,

雖然速度上的對比并不像上面的類比那樣明顯,但PyPy和CPython和這個道理一樣,

當使用PyPy運行腳本時,它會執行許多操作以使代碼運行得更快,如果腳本本身很簡單,則實際腳本運行速度會低于CPython,另一方面,如果您有一個長時間運行的腳本,那么可能會帶來顯著的性能提升,

想親自感受一下的話,請在CPython和PyPy中運行以下小腳本:

import time

start_time = time.time()

for i in range(100):
    print(i)

end_time = time.time()
print(f"It took {end_time-start_time:.10f} seconds to compute")

當您使用PyPy運行它時,開始時會有一個小延遲,而CPython會立即運行它,在Mac Book Pro上運行它,用CPython需要0.0004873276秒,用PyPy需要0.0019447803秒,

它不執行提前編譯

正如您在本教程開頭所看到的,PyPy不是一個完全編譯型的Python實作,它編譯Python代碼,但不是Python代碼的編譯器,由于Python固有的一些特性,導致無法將Python編譯為獨立的二進制檔案并重用它,

Py Py比完全解釋型的語言快,但比完全編譯的語言(如C)慢,

總結

PyPy是CPython的一種快速且功能強大的替代方案,使用它運行腳本,您可以在不更改代碼的情況下大大提高速度,但它也不是萬能的,有一些局限性,

在本教程中,您學習了:

  • PyPy是什么?

  • 如何安裝PyPy并使用它運行腳本

  • PyPy與CPython在速度方面的比較

  • PyPy的功能及其如何提高程式速

  • 在哪些情況下PyPy會有局限性

如果您的Python腳本需要稍微提高速度,歡迎嘗試PyPy!

長按掃碼添加“Python小助手” 一起討論PyPy!

▼點擊成為社區會員   喜歡就點個在看吧

轉載請註明出處,本文鏈接:https://www.uj5u.com/qianduan/217893.html

標籤:其他

上一篇:Android selinux知識點總結

下一篇:中國開源燃燒!Zilliz 獲全球開源基礎軟體最大單筆融資 4300 萬美元

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • vue移動端上拉加載

    可能做得過于簡單或者比較low,請各位大佬留情,一起探討技術 ......

    uj5u.com 2020-09-10 04:38:07 more
  • 優美網站首頁,頂部多層導航

    一個個人用的瀏覽器首頁,可以把一下常用的網站放在這里,平常打開會比較方便。 第一步,HTML代碼 <script src=https://www.cnblogs.com/szharf/p/"js/jquery-3.4.1.min.js"></script> <div id="navigate"> <ul> <li class="labels labels_1"> ......

    uj5u.com 2020-09-10 04:38:47 more
  • 頁面為要加<!DOCTYPE html>

    最近因為寫一個js函式,需要用到$(window).height(); 由于手寫demo的時候,過于自信,其實對前端方面的認識也不夠體系,用文本檔案直接敲出來的html代碼,第一行沒有加上<!DOCTYPE html> 導致了$(window).height();的結果直接是整個document的高 ......

    uj5u.com 2020-09-10 04:38:52 more
  • WordPress網站程式手動升級要做好資料備份

    WordPress博客網站程式在進行升級前,必須要做好網站資料的備份,這個問題良家佐言是遇見過的;在剛開始接觸WordPress博客程式的時候,因為升級問題和博客網站的修改的一些嘗試,良家佐言是吃盡了苦頭。因為購買的是西部數碼的空間和域名,每當佐言把自己的WordPress博客網站搞到一塌糊涂的時候 ......

    uj5u.com 2020-09-10 04:39:30 more
  • WordPress程式不能升級為5.4.2版本的原因

    WordPress是一款個人博客系統,受到英文博客愛好者和中文博客愛好者的追捧,并逐步演化成一款內容管理系統軟體;它是使用PHP語言和MySQL資料庫開發的,用戶可以在支持PHP和MySQL資料庫的服務器上使用自己的博客。每一次WordPress程式的更新,就會牽動無數WordPress愛好者的心, ......

    uj5u.com 2020-09-10 04:39:49 more
  • 使用CSS3的偽元素進行首字母下沉和首行改變樣式

    網頁中常見的一種效果,首字改變樣式或者首行改變樣式,效果如下圖。 代碼: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, ......

    uj5u.com 2020-09-10 04:40:09 more
  • 關于a標簽的講解

    什么是a標簽? <a> 標簽定義超鏈接,用于從一個頁面鏈接到另一個頁面。 <a> 元素最重要的屬性是 href 屬性,它指定鏈接的目標。 a標簽的語法格式:<a href=https://www.cnblogs.com/summerxbc/p/"指定要跳轉的目標界面的鏈接">需要展示給用戶看見的內容</a> a標簽 在所有瀏覽器中,鏈接的默認外觀如下: 未被訪問的鏈接帶 ......

    uj5u.com 2020-09-10 04:40:11 more
  • 前端輪播圖

    在需要輪播的頁面是引入swiper.min.js和swiper.min.css swiper.min.js地址: 鏈接:https://pan.baidu.com/s/15Uh516YHa4CV3X-RyjEIWw 提取碼:4aks swiper.min.css地址 鏈接:https://pan.b ......

    uj5u.com 2020-09-10 04:40:13 more
  • 如何設定html中的背景圖片(全屏顯示,且不拉伸)

    1 <style>2 body{background-image:url(https://uploadbeta.com/api/pictures/random/?key=BingEverydayWallpaperPicture); 3 background-size:cover;background ......

    uj5u.com 2020-09-10 04:40:16 more
  • Java學習——HTML詳解(上)

    HTML詳解 初識HTML Hyper Text Markup Language(超文本標記語言) 1 <!--DOCTYPE:告訴瀏覽器我們要使用什么規范--> 2 <!DOCTYPE html> 3 <html lang="en"> 4 <head> 5 <!--meta 描述性的標簽,描述一些 ......

    uj5u.com 2020-09-10 04:40:33 more
最新发布
  • 我的第一個NPM包:panghu-planebattle-esm(胖虎飛機大戰)使用說明

    好家伙,我的包終于開發完啦 歡迎使用胖虎的飛機大戰包!! 為你的主頁添加色彩 這是一個有趣的網頁小游戲包,使用canvas和js開發 使用ES6模塊化開發 效果圖如下: (覺得圖片太sb的可以自己改) 代碼已開源!! Git: https://gitee.com/tang-and-han-dynas ......

    uj5u.com 2023-04-20 07:59:23 more
  • 生產事故-走近科學之消失的JWT

    入職多年,面對生產環境,盡管都是小心翼翼,慎之又慎,還是難免捅出簍子。輕則滿頭大汗,面紅耳赤。重則系統停擺,損失資金。每一個生產事故的背后,都是寶貴的經驗和教訓,都是專案成員的血淚史。為了更好地防范和遏制今后的各類事故,特開此專題,長期更新和記錄大大小小的各類事故。有些是親身經歷,有些是經人耳傳口授 ......

    uj5u.com 2023-04-18 07:55:04 more
  • 記錄--Canvas實作打飛字游戲

    這里給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 打開游戲界面,看到一個畫面簡潔、卻又富有挑戰性的游戲。螢屏上,有一個白色的矩形框,里面不斷下落著各種單詞,而我需要迅速地輸入這些單詞。如果我輸入的單詞與螢屏上的單詞匹配,那么我就可以獲得得分;如果我輸入的單詞錯誤或者時間過長,那么我就會輸 ......

    uj5u.com 2023-04-04 08:35:30 more
  • 了解 HTTP 看這一篇就夠

    在學習網路之前,了解它的歷史能夠幫助我們明白為何它會發展為如今這個樣子,引發探究網路的興趣。下面的這張圖片就展示了“互聯網”誕生至今的發展歷程。 ......

    uj5u.com 2023-03-16 11:00:15 more
  • 藍牙-低功耗中心設備

    //11.開啟藍牙配接器 openBluetoothAdapter //21.開始搜索藍牙設備 startBluetoothDevicesDiscovery //31.開啟監聽搜索藍牙設備 onBluetoothDeviceFound //30.停止監聽搜索藍牙設備 offBluetoothDevi ......

    uj5u.com 2023-03-15 09:06:45 more
  • canvas畫板(滑鼠和觸摸)

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>canves</title> <style> #canvas { cursor:url(../images/pen.png),crosshair; } #canvasdiv{ bo ......

    uj5u.com 2023-02-15 08:56:31 more
  • 手機端H5 實作自定義拍照界面

    手機端 H5 實作自定義拍照界面也可以使用 MediaDevices API 和 <video> 標簽來實作,和在桌面端做法基本一致。 首先,使用 MediaDevices.getUserMedia() 方法獲取攝像頭媒體流,并將其傳遞給 <video> 標簽進行渲染。 接著,使用 HTML 的 < ......

    uj5u.com 2023-01-12 07:58:22 more
  • 記錄--短視頻滑動播放在 H5 下的實作

    這里給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 短視頻已經無數不在了,但是主體還是使用 app 來承載的。本文講述 H5 如何實作 app 的視頻滑動體驗。 無聲勝有聲,一圖頂百辯,且看下圖: 網址鏈接(需在微信或者手Q中瀏覽) 從上圖可以看到,我們主要實作的功能也是本文要講解的有: ......

    uj5u.com 2023-01-04 07:29:05 more
  • 一文讀懂 HTTP/1 HTTP/2 HTTP/3

    從 1989 年萬維網(www)誕生,HTTP(HyperText Transfer Protocol)經歷了眾多版本迭代,WebSocket 也在期間萌芽。1991 年 HTTP0.9 被發明。1996 年出現了 HTTP1.0。2015 年 HTTP2 正式發布。2020 年 HTTP3 或能正... ......

    uj5u.com 2022-12-24 06:56:02 more
  • 【HTML基礎篇002】HTML之form表單超詳解

    ??一、form表單是什么

    ??二、form表單的屬性

    ??三、input中的各種Type屬性值

    ??四、標簽 ......

    uj5u.com 2022-12-18 07:17:06 more