展開(kāi) 源起 今天和一新來(lái)的同事溝通,說(shuō)他用python編寫(xiě)了一個(gè)工程,但在第一次運(yùn)行后,發(fā)現(xiàn)工程根目錄下生成了一個(gè)__pycache__文件夾,里面是和py文件同名的各種以 .cpython-35.pyc 結(jié)尾的文件,問(wèn)同事都不太清楚,所以便抽空整理了一下該知識(shí)點(diǎn)。先解釋下cpython-35,cpython代表的是c語(yǔ)言實(shí)現(xiàn)的Python解釋器,-35代表的是版本為3.5版。 至于pyc,先來(lái)了解一下模塊的調(diào)用。 模塊的調(diào)用 Python中導(dǎo)入模塊時(shí),實(shí)際上會(huì)把被導(dǎo)入的模塊執(zhí)行一遍,如下: 先看被調(diào)用的模塊test.py: def haha(): print("哈哈") haha() 再看主程序main.py: import test print("一條大樹(shù)") 執(zhí)行結(jié)果是: 哈哈 一條大樹(shù) 那怎么才能只是單純調(diào)用而不執(zhí)行被調(diào)用模塊的代碼呢?要想被調(diào)用模塊代碼不被執(zhí)行,前提得知道變量__name__是什么意思,簡(jiǎn)單來(lái)說(shuō)就是,如果不涉及模塊導(dǎo)入的話,__name__的值就是” __main__“,如果當(dāng)此模塊被導(dǎo)入引用的話,那么這個(gè)模塊內(nèi)的__name__值就是文件的名字(不帶.py),如下test_1.py: def haha(): print("哈哈") haha() print(__name__) test_1.py執(zhí)行結(jié)果為: 哈哈 __main__ 如果test_1被導(dǎo)入引用的話,如test_2: import test_1 print("一條大樹(shù)") test_2x運(yùn)行結(jié)果為: 哈哈 test_1 一條大樹(shù) 上邊所說(shuō)要是弄懂的話,那我們?cè)诒徽{(diào)用的模塊中,可執(zhí)行的代碼前加上這么一句判斷,if __name__ == '__main__':,被調(diào)用的模塊的代碼就不會(huì)被執(zhí)行了! 接下來(lái)才是正題 以下參考自Joy_Shen的一個(gè)回答。 先大概了解一下python基本運(yùn)行機(jī)制。Python程序運(yùn)行時(shí)不需要編譯成二進(jìn)制代碼,而直接從源碼運(yùn)行程序,簡(jiǎn)單來(lái)說(shuō)是,Python解釋器將源碼轉(zhuǎn)換為字節(jié)碼,然后再由解釋器來(lái)執(zhí)行這些字節(jié)碼。 解釋器的具體工作: 1、完成模塊的加載和鏈接; 2、將源代碼編譯為PyCodeObject對(duì)象(即字節(jié)碼),寫(xiě)入內(nèi)存中,供CPU讀??; 3、從內(nèi)存中讀取并執(zhí)行,結(jié)束后將PyCodeObject寫(xiě)回硬盤(pán)當(dāng)中,也就是復(fù)制到.pyc或.pyo文件中,以保存當(dāng)前目錄下所有腳本的字節(jié)碼文件。 之后若再次執(zhí)行該腳本,它先檢查【本地是否有上述字節(jié)碼文件】和【該字節(jié)碼文件的修改時(shí)間是否在其源文件之后】,是就直接執(zhí)行,否則重復(fù)上述步驟。 那有的小伙伴就有疑問(wèn)了,__pycache__文件夾的意義何在呢? 因?yàn)榈谝淮螆?zhí)行代碼的時(shí)候,Python解釋器已經(jīng)把編譯的字節(jié)碼放在__pycache__文件夾中,這樣以后再次運(yùn)行的話,如果被調(diào)用的模塊未發(fā)生改變,那就直接跳過(guò)編譯這一步,直接去__pycache__文件夾中去運(yùn)行相關(guān)的 *.pyc 文件,大大縮短了項(xiàng)目運(yùn)行前的準(zhǔn)備時(shí)間。
|
|