
Python中模塊與包有相同名字的處理方法
在編程開(kāi)發(fā)中,個(gè)人覺(jué)得,只要按照規(guī)范去做,很少會(huì)出問(wèn)題。剛開(kāi)始學(xué)習(xí)一門(mén)技術(shù)時(shí),的確會(huì)遇到很多的坑。踩的坑多了,這是好事,會(huì)學(xué)到更多東西,也會(huì)越來(lái)越覺(jué)得按照規(guī)范做的重要性,規(guī)范的制定就是用來(lái)規(guī)避問(wèn)題的。有時(shí)候確實(shí)應(yīng)該聽(tīng)聽(tīng)有經(jīng)驗(yàn)人的建議,不要一意孤行。這好像不是本文的重點(diǎn),其實(shí)我重點(diǎn)是想表達(dá),盡量按規(guī)范做事,這樣會(huì)少走很多彎路。
我現(xiàn)在使用的主力編程語(yǔ)言是 Python,在接觸 Python 至今,我感覺(jué)我踩的坑還是極少的,基本上沒(méi)有遇到什么奇怪的問(wèn)題。實(shí)際上,這并不是一件好事,不踩坑,很多躺在暗處的知識(shí)點(diǎn)你不會(huì)了解,所以也很難成長(zhǎng)。幸好,有一些會(huì)踩坑的同事。
一同事問(wèn)我,在 Python 中,如果一個(gè)模塊和一個(gè)包同名時(shí),是不是只能導(dǎo)入包,如果要導(dǎo)入模塊該怎么辦。他的意思大概是這樣的,在項(xiàng)目的同一級(jí)目錄下,有一個(gè) foo.py 文件和一個(gè) foo/ 目錄,如果 import foo 會(huì)導(dǎo)入 foo/ 的內(nèi)容而不是 foo.py 的內(nèi)容。
被問(wèn)到這個(gè)問(wèn)題時(shí),我首先感覺(jué)到的是詫異,這明顯是存在歧義的。如果是我,肯定不會(huì)把模塊名和包名設(shè)計(jì)成一樣的名字,因?yàn)楸举|(zhì)上來(lái)說(shuō)在導(dǎo)入的時(shí)候沒(méi)法區(qū)分到底要導(dǎo)入誰(shuí)。除非系統(tǒng)有特別的規(guī)定,例如,規(guī)定這種情況只能導(dǎo)入包。
我的潛意識(shí)里認(rèn)為這里應(yīng)該報(bào)錯(cuò),Python 解釋器不知道要導(dǎo)入誰(shuí)。但是,同事告訴我,別人的代碼是這么寫(xiě)的,而且在這種情況下會(huì)默認(rèn)導(dǎo)入包。那就是可以的咯,而且解釋器已經(jīng)規(guī)定這種情況會(huì)總是導(dǎo)入包。
為了驗(yàn)證下這一點(diǎn),我寫(xiě)了個(gè)簡(jiǎn)單的項(xiàng)目,項(xiàng)目結(jié)構(gòu)如下:
.
├── main.py
└── same
├── api
│ └── __init__.py
├── auth
│ └── __init__.py
├── auth.py
└── __init__.py
其中:
same/api/__init__/py 的內(nèi)容:
from .. import auth
same/auth/__init__.py 的內(nèi)容:
auth_str = "This is str in package!"
same/auth.py 的內(nèi)容:
auth_str = "This is str in module!"
main.py 的內(nèi)容:
from __future__ import print_function
from same.api import auth
# Script starts from here
if __name__ == "__main__":
print(auth.auth_str)
稍微有些復(fù)雜,哈哈,主要是同事那兒大致的結(jié)構(gòu)是這樣的,這里是為更好的模擬下。我在 same.auth 包中定義了一個(gè) auth_str
字符串,又在同名的 same.auth 模塊中定義了一個(gè)同名的 auth_str 字符串,然后在 same.api 包嘗試導(dǎo)入 auth,最后在
main.py 嘗試輸出 same.api.auth.auth_str,看看到底哪個(gè)字符串會(huì)被打印。同時(shí)嘗試用 Python2 和
Python3 執(zhí)行 main.py,得到的結(jié)果都是:
This is str in package!
這里驗(yàn)證了我們的猜想是正確的,解釋器的確只導(dǎo)入了包中內(nèi)容。但是,我并不知道是否有官方的資料說(shuō)明就是這樣的,所以我不敢確信,萬(wàn)一這只是巧合呢。
于是,我開(kāi)始查資料來(lái)驗(yàn)證這一結(jié)論。我就說(shuō)實(shí)話吧,對(duì)于一個(gè)英文水平爛到你無(wú)法想象的我,只能先嘗試用百度搜索下答案了。事實(shí)是,用百度往往都是遺憾的。片刻后,無(wú)果,我只能硬著頭皮嘗試英文搜索了。于是,在 stackoverflow 上找到了如下提問(wèn):
How python deals with module and package having the same name?
其中有一個(gè)人回答說(shuō) Python 官方文檔中在描述模塊搜索路徑時(shí)提到了這一點(diǎn):https://docs.python.org/3/tutorial/modules.html#the-module-search-path.
文檔中有如下一段描述:
After initialization, Python programs can modify sys.path. The
directory containing the script being run is placed at the beginning of
the search path, ahead of the standard library path. This means that
scripts in that directory will be loaded instead of modules of the same
name in the library directory. This is an error unless the replacement
is intended. See section Standard Modules for more information.
也就是說(shuō),目錄在庫(kù)的搜索路徑下會(huì)首先被搜索,這就意味著目錄會(huì)代替同名的模塊被加載。
這下終于放心了,之前的結(jié)論得到證實(shí)。在 Python 中,如果嘗試導(dǎo)入同名的模塊和包時(shí),包會(huì)被導(dǎo)入。這種情況下,如果想要導(dǎo)入模塊,恐怕要用一些
‘hack' 的方法,上面提到的 stackoverflow
帖下有一些示例可以參考。當(dāng)然,最好的方法是避免這樣的設(shè)計(jì),這樣你就不會(huì)花那么長(zhǎng)時(shí)間去查資料,也不會(huì)花那么長(zhǎng)時(shí)間來(lái)寫(xiě)類似于本文的文章。
數(shù)據(jù)分析咨詢請(qǐng)掃描二維碼
若不方便掃碼,搜微信號(hào):CDAshujufenxi
SQL Server 中 CONVERT 函數(shù)的日期轉(zhuǎn)換:從基礎(chǔ)用法到實(shí)戰(zhàn)優(yōu)化 在 SQL Server 的數(shù)據(jù)處理中,日期格式轉(zhuǎn)換是高頻需求 —— 無(wú)論 ...
2025-09-18MySQL 大表拆分與關(guān)聯(lián)查詢效率:打破 “拆分必慢” 的認(rèn)知誤區(qū) 在 MySQL 數(shù)據(jù)庫(kù)管理中,“大表” 始終是性能優(yōu)化繞不開(kāi)的話題。 ...
2025-09-18CDA 數(shù)據(jù)分析師:表結(jié)構(gòu)數(shù)據(jù) “獲取 - 加工 - 使用” 全流程的賦能者 表結(jié)構(gòu)數(shù)據(jù)(如數(shù)據(jù)庫(kù)表、Excel 表、CSV 文件)是企業(yè)數(shù)字 ...
2025-09-18DSGE 模型中的 Et:理性預(yù)期算子的內(nèi)涵、作用與應(yīng)用解析 動(dòng)態(tài)隨機(jī)一般均衡(Dynamic Stochastic General Equilibrium, DSGE)模 ...
2025-09-17Python 提取 TIF 中地名的完整指南 一、先明確:TIF 中的地名有哪兩種存在形式? 在開(kāi)始提取前,需先判斷 TIF 文件的類型 —— ...
2025-09-17CDA 數(shù)據(jù)分析師:解鎖表結(jié)構(gòu)數(shù)據(jù)特征價(jià)值的專業(yè)核心 表結(jié)構(gòu)數(shù)據(jù)(以 “行 - 列” 規(guī)范存儲(chǔ)的結(jié)構(gòu)化數(shù)據(jù),如數(shù)據(jù)庫(kù)表、Excel 表、 ...
2025-09-17Excel 導(dǎo)入數(shù)據(jù)含缺失值?詳解 dropna 函數(shù)的功能與實(shí)戰(zhàn)應(yīng)用 在用 Python(如 pandas 庫(kù))處理 Excel 數(shù)據(jù)時(shí),“缺失值” 是高頻 ...
2025-09-16深入解析卡方檢驗(yàn)與 t 檢驗(yàn):差異、適用場(chǎng)景與實(shí)踐應(yīng)用 在數(shù)據(jù)分析與統(tǒng)計(jì)學(xué)領(lǐng)域,假設(shè)檢驗(yàn)是驗(yàn)證研究假設(shè)、判斷數(shù)據(jù)差異是否 “ ...
2025-09-16CDA 數(shù)據(jù)分析師:掌控表格結(jié)構(gòu)數(shù)據(jù)全功能周期的專業(yè)操盤(pán)手 表格結(jié)構(gòu)數(shù)據(jù)(以 “行 - 列” 存儲(chǔ)的結(jié)構(gòu)化數(shù)據(jù),如 Excel 表、數(shù)據(jù) ...
2025-09-16MySQL 執(zhí)行計(jì)劃中 rows 數(shù)量的準(zhǔn)確性解析:原理、影響因素與優(yōu)化 在 MySQL SQL 調(diào)優(yōu)中,EXPLAIN執(zhí)行計(jì)劃是核心工具,而其中的row ...
2025-09-15解析 Python 中 Response 對(duì)象的 text 與 content:區(qū)別、場(chǎng)景與實(shí)踐指南 在 Python 進(jìn)行 HTTP 網(wǎng)絡(luò)請(qǐng)求開(kāi)發(fā)時(shí)(如使用requests ...
2025-09-15CDA 數(shù)據(jù)分析師:激活表格結(jié)構(gòu)數(shù)據(jù)價(jià)值的核心操盤(pán)手 表格結(jié)構(gòu)數(shù)據(jù)(如 Excel 表格、數(shù)據(jù)庫(kù)表)是企業(yè)最基礎(chǔ)、最核心的數(shù)據(jù)形態(tài) ...
2025-09-15Python HTTP 請(qǐng)求工具對(duì)比:urllib.request 與 requests 的核心差異與選擇指南 在 Python 處理 HTTP 請(qǐng)求(如接口調(diào)用、數(shù)據(jù)爬取 ...
2025-09-12解決 pd.read_csv 讀取長(zhǎng)浮點(diǎn)數(shù)據(jù)的科學(xué)計(jì)數(shù)法問(wèn)題 為幫助 Python 數(shù)據(jù)從業(yè)者解決pd.read_csv讀取長(zhǎng)浮點(diǎn)數(shù)據(jù)時(shí)的科學(xué)計(jì)數(shù)法問(wèn)題 ...
2025-09-12CDA 數(shù)據(jù)分析師:業(yè)務(wù)數(shù)據(jù)分析步驟的落地者與價(jià)值優(yōu)化者 業(yè)務(wù)數(shù)據(jù)分析是企業(yè)解決日常運(yùn)營(yíng)問(wèn)題、提升執(zhí)行效率的核心手段,其價(jià)值 ...
2025-09-12用 SQL 驗(yàn)證業(yè)務(wù)邏輯:從規(guī)則拆解到數(shù)據(jù)把關(guān)的實(shí)戰(zhàn)指南 在業(yè)務(wù)系統(tǒng)落地過(guò)程中,“業(yè)務(wù)邏輯” 是連接 “需求設(shè)計(jì)” 與 “用戶體驗(yàn) ...
2025-09-11塔吉特百貨孕婦營(yíng)銷案例:數(shù)據(jù)驅(qū)動(dòng)下的精準(zhǔn)零售革命與啟示 在零售行業(yè) “流量紅利見(jiàn)頂” 的當(dāng)下,精準(zhǔn)營(yíng)銷成為企業(yè)突圍的核心方 ...
2025-09-11CDA 數(shù)據(jù)分析師與戰(zhàn)略 / 業(yè)務(wù)數(shù)據(jù)分析:概念辨析與協(xié)同價(jià)值 在數(shù)據(jù)驅(qū)動(dòng)決策的體系中,“戰(zhàn)略數(shù)據(jù)分析”“業(yè)務(wù)數(shù)據(jù)分析” 是企業(yè) ...
2025-09-11Excel 數(shù)據(jù)聚類分析:從操作實(shí)踐到業(yè)務(wù)價(jià)值挖掘 在數(shù)據(jù)分析場(chǎng)景中,聚類分析作為 “無(wú)監(jiān)督分組” 的核心工具,能從雜亂數(shù)據(jù)中挖 ...
2025-09-10統(tǒng)計(jì)模型的核心目的:從數(shù)據(jù)解讀到?jīng)Q策支撐的價(jià)值導(dǎo)向 統(tǒng)計(jì)模型作為數(shù)據(jù)分析的核心工具,并非簡(jiǎn)單的 “公式堆砌”,而是圍繞特定 ...
2025-09-10