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