python關於多級包之間的引用問題

西西嘛呦發表於2021-08-22

首先得明確包和模組。
:在一個目錄下存在__init__.py,那麼該目錄就是一個包。
模組:一個.py檔案就是一個模組。
我們可以通過from 包 import 模組來引入python檔案,
也可以使用from 包.模組 import 模組中的函式或類。
具體看一下例子。
假設我們現在有以下的目錄:
image
我們想在main.py中使用package_a和package_b裡面額模組,可以這麼使用:

from package_a import tmp2
from package_b import tmp4

tmp2.test_aTmp2()
tmp4.test_bTmp4()

test_aTmp2()和test_bTmp4()是tmp2.py和tmp4.py裡面的函式。

這裡是package_a包中tmp2模組下的test_aTmp2函式
這裡是package_b包中tmp4模組下的test_bTmp4函式

假設我們想在main.py中使用a_utils包下的aUtils模組和b_utils包下的bUtils模組,我們可以這樣:

from package_a import tmp2
from package_b import tmp4
from package_b.b_utils.bUtils import test_bUtils
from package_a.a_utils import aUtils

tmp2.test_aTmp2()
tmp4.test_bTmp4()
aUtils.test_aUtils()
test_bUtils()

注意這裡的兩種用法,

  • 一種是我們直接通過:from 包.包.模組 import 函式
  • 另一種是from 包.包 import 模組,並通過 模組.函式 來使用相關的函式。
這裡是package_a包中tmp2模組下的test_aTmp2函式
這裡是package_b包中tmp4模組下的test_bTmp4函式
這裡是package_a包中a_uitls包中的aUtils模組下的test_aUitls函式
這裡是package_b包中b_uitls包中的bUtils模組下的test_buitls函式

下一個,假設我們想在tmp4.py使用tmp3.py中的函式,因為在同一個包下,我們自然的是這麼使用:

from tmp3 import test_bTmp3
test_bTmps3()

def test_bTmp4():
    print('這裡是package_b包中tmp4模組下的test_bTmp4函式')

我們在tmp4.py中繼續使用b_uitls包下的bUtils.py。

from tmp3 import test_bTmp3
from b_utils.bUtils import test_bUtils

def test_bTmp4():
    print('這裡是package_b包中tmp4模組下的test_bTmp4函式')

# test_bUtils()
test_bTmp3()
test_bUtils()
這裡是package_b包中tmp3模組下的test_bTmp3函式
這裡是package_b包中b_uitls包中的bUtils模組下的test_buitls函式

現在的tmp4.py裡面是以上那樣,假設我們現在重新執行之前的main.py:

from package_a import tmp2
from package_b import tmp4
from package_b.b_utils.bUtils import test_bUtils
from package_a.a_utils import aUtils

tmp2.test_aTmp2()
tmp4.test_bTmp4()
aUtils.test_aUtils()
test_bUtils()

你會發現報錯了:

Traceback (most recent call last):
  File "C:/Users/龔鷗波/Desktop/test_import/main.py", line 2, in <module>
    from package_b import tmp4
  File "C:\Users\龔鷗波\Desktop\test_import\package_b\tmp4.py", line 8, in <module>
    from tmp3 import test_bTmp3
ModuleNotFoundError: No module named 'tmp3'

我們在main.py裡面匯入了tmp4,tmp4裡面呼叫了tmp3,我門執行tmp4.py是沒問題,執行main.py有問題。這是因為,相對於main.py所在的目錄,tmp3所在是在包package_b下面,在tmp4中就不能直接匯入,可以改寫成以下方式:

from .tmp3 import test_bTmp3
from .b_utils.bUtils import test_bUtils

def test_bTmp4():
    print('這裡是package_b包中tmp4模組下的test_bTmp4函式')

test_bTmp3()
test_bUtils()

這樣就不會報錯了:

這裡是package_b包中tmp3模組下的test_bTmp3函式
這裡是package_b包中b_uitls包中的bUtils模組下的test_buitls函式
這裡是package_a包中tmp2模組下的test_aTmp2函式
這裡是package_b包中tmp4模組下的test_bTmp4函式
這裡是package_a包中a_uitls包中的aUtils模組下的test_aUitls函式
這裡是package_b包中b_uitls包中的bUtils模組下的test_buitls函式

但是這時我們去重新執行tmp4.py,發現會報錯了:

Traceback (most recent call last):
  File "C:/Users/龔鷗波/Desktop/test_import/package_b/tmp4.py", line 8, in <module>
    from .tmp3 import test_bTmp3
ModuleNotFoundError: No module named '__main__.tmp3'; '__main__' is not a package

這裡.表示的是__main__,我暫時也不清楚這時是什麼意思,不過我們可以這麼改寫下:

try:
    from tmp3 import test_bTmp3
    from b_utils.bUtils import test_bUtils
except Exception as e:
    from .tmp3 import test_bTmp3
    from .b_utils.bUtils import test_bUtils

def test_bTmp4():
    print('這裡是package_b包中tmp4模組下的test_bTmp4函式')

test_bTmp3()
test_bUtils()

這樣不論我們是執行main.py還是tmp4.py就都不會報錯了。
以上是自己在建專案中碰到的一個問題。

相關文章