佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > 淺析Python模塊之間的相互引用問題

淺析Python模塊之間的相互引用問題

熱門標簽:語音系統(tǒng) 電話運營中心 呼叫中心市場需求 Win7旗艦版 硅谷的囚徒呼叫中心 企業(yè)做大做強 客戶服務 百度AI接口

摘要:詳細講解了相對路徑和絕對路徑的引用方法。

在某次運行過程中出現(xiàn)了如下兩個報錯:

報錯1: ModuleNotFoundError: No module named '__main__.src_test1'; '__main__' is not a package
報錯2: ImportError: attempted relative import with no known parent package

于是基于這兩個報錯探究了一下python3中的模塊相互引用的問題,下面來逐個解析,請耐心看完。

好的,我們先來構造第一個錯,測試代碼結構如下:

|--- test_main.py
|--- src
 |--- __init__.py                               
  |--- src_test1.py
  |--- src_test2.py

src_test2.py 代碼

class Test2(object):
  def foo(self):
    print('I am foo')

src_test1.py 代碼,引用Test2模塊

from .src_test2 import Test2
 
def fun1():
  t2 = Test2()
  t2.foo()
if __name__ == "__main__":
  fun1()

此時運行 src_test1.py 報錯“No module named '__main__.src_test1'; '__main__' is not a package”

問題原因:

主要在于引用src_test2模塊的時候,用的是相對路徑".",在import語法中翻譯成"./",也就是當前目錄下,按這樣理解也沒有問題,那為什么報錯呢?

從 PEP 328 中,我們找到了關于 the relative imports(相對引用)的介紹

通俗一點意思就是,你程序入口運行的那個模塊,就默認為主模塊,他的name就是‘main',然后會將本模塊import中的點(.)替換成‘__main__',那么 .src_test2就變成了 __main__.src_test2,所以當然找不到這個模塊了。

解決方法:

因此,建議的做法是在 src同層級目錄創(chuàng)建 引用模塊 test_main.py(為什么不在src目錄下創(chuàng)建,待會下一個報錯再講),并引用src_test1模塊,代碼如下:

from src.src_test1 import fun1
 
if __name__ == "__main__":
  fun1()

那為什么這樣執(zhí)行就可以了呢,其中原理是什么呢?我是這樣理解的(歡迎糾正):test_main執(zhí)行時,他被當做根目錄,因此他引用的src.src_test1 是絕對路徑,這樣引用到哪都不會錯,此時他的name=‘main',當執(zhí)行src_test1的時候,注意了此時test1的name是 src.src_test1,那么在test1中使用的是相對路徑,查找邏輯是先找到父節(jié)點(src目錄),再找父節(jié)點下面的src_test2,因此可以成功找到,Bingo!

輔證:

構造一個例子,就可以理解上面的 執(zhí)行目錄就是根目錄 的說法了,修改test1,使引用test_main:

from .. import test_main
 
報錯:ValueError: attempted relative import beyond top-level package

OK,那繼續(xù)構造第二個報錯:

上文中說過,解決main 的問題,就是創(chuàng)建一個模塊,來調(diào)用使用相對路徑的模塊,那么為什么我不能在相同目錄下創(chuàng)建這個文件來調(diào)用呢?讓我們來測試下代碼:

創(chuàng)建test_src.py文件,代碼結構變更如下:

|--- test_main.py
|--- src
 |--- __init__.py                               
  |--- src_test1.py
  |--- src_test2.pys
  |--- test_src.py

test_src 代碼:

from src_test1 import fun1
 
if __name__ == "__main__":
  fun1()

執(zhí)行報錯:ImportError: attempted relative import with no known parent package

問題原因:

當執(zhí)行test_src時,按上文理解,此時執(zhí)行文件所在的目錄為根目錄,那么引用test1的時候,需要注意的是,此時test1的name屬性不再是src.src_test1,因為程序感知不到src的存在,此時他的絕對路徑是 src_test1,此時再次引用相對路徑查找的test2,同樣的步驟,需要先找到父節(jié)點,而此時他自己就是根節(jié)點了,已經(jīng)沒有父節(jié)點了,因此報錯“no known parent package”。

解決方法:

此時為了避免父節(jié)點產(chǎn)生矛盾,因此將test1中的引入去掉相對引用即可

from .src_test2 import Test2  -->  from src_test2 import Test2

繼續(xù)深入:

那使用相對路徑和絕對路徑,編譯器是怎么找到這個模塊的呢?

執(zhí)行import的時候,存在一個引入的順序,即優(yōu)先查找執(zhí)行目錄下有沒有此文件,如沒有,再查找lib庫下,如還沒有,再查找sys.path中的路徑,如再沒有,報錯。

所以不管是當前目錄,還是 sys.path中的目錄,都可以查到 src_test2這個模塊,就可以編譯成功。

號外:

解決完上述問題后,不管我們用哪種方式,我們調(diào)試代碼時,都是單個文件調(diào)試,但此時根目錄就不對了,import方式又要改動,執(zhí)行起來很麻煩,所以這里推薦另一種方式(有更好的方式歡迎留言),使用sys.path.append()的方法

import sys,os
sys.path.append(os.getcwd())
from src.src_test2 import Test2

使用append的方式,將程序文件根目錄放進了sys.path中,然后再引用絕對路徑,這樣的方式,不管使用上文中的第一或第二執(zhí)行方式都可以調(diào)用,也可以單獨編譯test1文件,不用修改import路徑,也是相對安全的方式。但是缺點就是,如果你修改了某一個包名,需要將所有引用地方都修改一下,工作量大,所以因地制宜。

綜上,詳細講解了相對路徑和絕對路徑的引用方法,現(xiàn)在你應該對import導入的問題有了清晰的理解吧

備注:本文基于Python3.7版本測試

到此這篇關于Python模塊之間的相互引用問題的文章就介紹到這了,更多相關Python模塊引用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • Python re.sub 反向引用的實現(xiàn)
  • 用Python代碼自動生成文獻的IEEE引用格式的實現(xiàn)
  • python3爬蟲中引用Queue的實例講解
  • python閉包與引用以及需要注意的陷阱
  • 詳解python如何引用包package
  • python 引用傳遞和值傳遞詳解(實參,形參)
  • Python參數(shù)傳遞機制傳值和傳引用原理詳解
  • Python參數(shù)傳遞對象的引用原理解析
  • python實現(xiàn)引用其他路徑包里面的模塊
  • python關于多級包之間的引用問題

標簽:海南 山西 安康 山西 長沙 崇左 喀什 濟南

巨人網(wǎng)絡通訊聲明:本文標題《淺析Python模塊之間的相互引用問題》,本文關鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    亚东县| 金门县| 平乡县| 益阳市| 双柏县| 沿河| 陆良县| 绥宁县| 洞头县| 手游| 开阳县| 汉寿县| 柏乡县| 安溪县| 宣化县| 简阳市| 桐乡市| 莱阳市| 万山特区| 寿阳县| 颍上县| 仲巴县| 鸡泽县| 花莲县| 精河县| 镇雄县| 察隅县| 隆子县| 紫金县| 威远县| 濮阳市| 台州市| 麻阳| 茌平县| 明水县| 礼泉县| 中方县| 凤城市| 东明县| 龙南县| 乐陵市|