在我們平常工程裡使用Python的過程中,經常需要解決各個模組的匯入問題,而且也常常遇到參照路徑查詢不到、交叉匯入模組等等問題,故寫這篇文章,旨在講述Python的模組匯入機制和我們平時大型專案中應該遵循的模組匯入規範
日常程式設計中,為了能夠複用寫過的程式碼邏輯,我們都會把這些程式碼封裝成為模組,需要用到的時候可以直接匯入複用,以便提高我們的開發效率。 module能定義函數、類、變數,也能包含可執行的程式碼。module來源有3種: ①Python內建的模組(標準庫); ②第三方模組; ③自定義模組;
模組的匯入一般是在檔案頭使用import關鍵字,import一個模組相當於先執行了一次這個被匯入模組,然後在本名稱空間建立一個與被匯入模組名稱空間的聯絡,相當於在本名稱空間新建了一個變數,這個變數名稱是被匯入模組的名稱,指向被匯入模組的名稱空間。所以匯入的這個模組相當於一個變數,因此多次匯入同一個模組只有第一次匯入的時候會被執行(後續匯入會判斷到這個模組變數已存在所以不執行)
每一個匯入的模組都會在Python內建字典sys.modules中,Python一啟動,它將被載入在記憶體中,當我們匯入新modules,sys.modules將自動記錄下該module。 Python的模組查詢路徑的機制是:
所以對於我們自己編寫的模組,如果封裝並行布到了PyPi,則可以用pip install直接安裝,並在啟動時載入在記憶體中,通過sys.modules可以檢視到 而對於僅需要在本專案中複用的模組,我們在複用程式碼中將其路徑加入到sys.path中,同樣可以參照到該模組。
所有的模組import都從「根節點」開始。根節點的位置由sys.path中的路徑決定,專案的根目錄一般自動在sys.path中。如果希望程式能處處執行,需手動修改sys.path
import sys,os BASE_DIR = os.path.dirname(os.path.abspath(__file__))#專案根目錄所在的絕對路徑sys.path.append(BASE_DIR)import A, B #匯入A、B包複製程式碼
只關心相對自己當前目錄的模組位置就好。不能在包(package)的內部直接執行(會報錯)。不管根節點在哪兒,包內的模組相對位置都是正確的。
#from . import b2 #這種匯入方式會報錯,只有在包內部直接執行的時候才可以這樣匯入。import b2#正確b2.print_b2()複製程式碼
當一個資料夾下有init.py時,意為該資料夾是一個包(package),其下的多個模組(module)構成一個整體,而這些模組(module)都可通過同一個包(package)匯入其他程式碼中。 其中init.py檔案 用於組織包(package),方便管理各個模組之間的參照、控制著包的匯入行為。
該檔案可以什麼內容都不寫,即為空檔案(為空時,僅僅用import [該包]形式 是什麼也做不了的),存在即可,相當於一個標記。
在python3中,即使包下沒有init.py檔案,import 包仍然不會報錯,而在python2中,包下一定要有該檔案,否則import 包會報錯
all 是一個重要的變數,用來指定此包(package)被import *時,哪些模組(module)會被import進【當前作用域中】。不在all列表中的模組不會被其他程式參照。可以重寫all,如 all= [‘當前所屬包模組1名字’, ‘模組1名字’],如果寫了這個,則會按列表中的模組名進行匯入
在包內部直接執行時,包的name == 'main',但是在外部匯入包是,可以通過
if __name__ == '__main__':複製程式碼
來避免實現包內部偵錯時的邏輯
當兩個模組A和B之間相互import時,就會出現迴圈匯入的問題,此時程式執行會報錯:can not import name xxx,如:
# a.pyprint('from a.py')from b import x y = 'a'複製程式碼
# b.pyprint('from b.py')from a import y x = 'b'複製程式碼
我們來分析一下這種錯誤是怎麼出現的:
因此在a.py中執行from b import x的順序就是1->3,先引入b,b裡面from a import y由相當於執行了a.py,順序是1->2,因為此時b已經引入所以不會執行3,2中無法找到x物件,因為引入b時還沒執行到x='b'這一步,所以報錯了
分離模組,將同一類別的模組放在同一目錄下,形成類別分明的目錄架構,如:
更多相關免費學習推薦:
以上就是學習Python模組匯入機制與大型專案的規範的詳細內容,更多請關注TW511.COM其它相關文章!