如何避免Python的循環(huán)導(dǎo)入問題

本文首載于Gevin的博客
未經(jīng)Gevin授權(quán),禁止轉(zhuǎn)載
原文鏈接:如何避免Python的循環(huán)導(dǎo)入問題

how to avoid python circle import error

Python 中使用package時(shí),出現(xiàn)循環(huán)導(dǎo)入問題十分常見,我們創(chuàng)建如下package來說明這個(gè)問題:

pkg
  ├── __init__.py
  ├── module_a.py
  └── module_b.py

其中,

  • __init__.pypkg指定為一個(gè)Python package
  • module_a.py中定義了一個(gè)action_a()函數(shù),該函數(shù)引用了module_b.py中的一個(gè)attribute,如一個(gè)函數(shù)或變量
  • module_b.py中定義了一個(gè)action_b()函數(shù),該函數(shù)引用了module_a.py中的一個(gè)attribute,如一個(gè)函數(shù)或變量

這種情況下,執(zhí)行該package時(shí)會(huì)拋出circular import error錯(cuò)誤,即循環(huán)引用,因?yàn)?code>module_a試圖去引入module_b時(shí),而module_b首先要引入module_a,這會(huì)導(dǎo)致Python解釋器無法執(zhí)行下去。

然而,我們可以通過一些巧妙的方法,讓上面的邏輯正常工作,同時(shí)避免循環(huán)引入的錯(cuò)誤。

那么,什么時(shí)候它能正常工作,什么時(shí)候不能正常工作,而那些能夠正常工作的情況又是什么原因呢?

何時(shí)它能正常工作?

1. 在module頂部引入,不要用from,相對引入,只在Python 2中有效

在module的頂部import,如import another_module,module 中的函數(shù)以another_module.attribute的方式引用another_module中的函數(shù)或變量等。這種方式之所以有效,是由于import another_module是基于當(dāng)前目錄的相對引用,而且是一種隱式引用,如果從另一個(gè)package中引入module時(shí),就可以失效了。另外,import another_module這種語法在Python 3 中已經(jīng)不支持了,所以不要在代碼中用這種方法來避免循環(huán)引入。

如:

# pkg/module_a.py 
from __future__ import print_function
import module_b

def action_a():
    print(module_b.action_b.__name__)


# pkg/module_b.py
from __future__ import print_function
import module_a

def action_b():
    print(module_a.action_a.__name__)

2. 在module的頂部引入,不要用from,絕對引入

在module的頂部import,使用從package開始的絕對路徑,如import package.another_module,module 中的函數(shù)以package.another_module.attribute的方式引用another_module中的函數(shù)或變量等。之所以要掛上package name來引入,是由于import .another_module這種形式的“相對引入”會(huì)報(bào)語法錯(cuò)誤,而掛上package的絕對引入,Python 2和3都支持

案例:

# pkg/module_a.py
from __future__ import print_function
import pkg2.module_b

def action_a():
    print(pkg2.module_b.action_b.__name__)


# pkg/module_b.py
from __future__ import print_function
import pkg2.module_a

def action_b():
    print(pkg2.module_a.action_a.__name__)

    

3. 在module底部引入another module的attribute,而非another module,用from

在module的底部import(至少要在被引用的attribute之后import),直接引入another module的attribute,如from package.another_module import attribute,相對引入也支持,如from .another_module import attribute,module中的函數(shù)直接使用被引用的attribute即可。

如:

# pkg/module_a.py
from __future__ import print_function

def action_a():
    print(action_b.__name__)

from .module_b import action_b


# pkg/module_b.py
from __future__ import print_function

def action_b():
    print(action_a.__name__)

from .module_a import action_a

4. 函數(shù)頂部引入,可以用from

在module的function頂部import,如from package import another_module,也支持相對引入,引入module或attribute均可。

如:

# pkg/module_a.py
from __future__ import print_function

def action_a():
    from . import module_b
    print(module_b.action_b.__name__)


# pkg/module_b.py
from __future__ import print_function

def action_b():
    from . import module_a
    print(module_a.action_a.__name__)

# pkg/module_a.py
from __future__ import print_function

def action_a():
    from .module_b import action_b
    print(action_b.__name__)


# pkg/module_b.py
from __future__ import print_function
def action_b():
    from .module_a import action_a
    print(action_a.__name__)


這種方式雖然Python 2和3都支持,但編碼不夠優(yōu)雅,影響代碼可讀性,不建議使用

  • 本文討論的問題,是Python中調(diào)用package時(shí),應(yīng)如何避免循環(huán)引入
  • 當(dāng)直接在命令行執(zhí)行一個(gè)Python module時(shí),適用情況不完全相同
  • 本文內(nèi)容我在GitHub上提供了一個(gè)Demo,歡迎查看或fork
  • Reference:This Gist
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 個(gè)人筆記,方便自己查閱使用 Py.LangSpec.Contents Refs Built-in Closure ...
    freenik閱讀 67,949評論 0 5
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,578評論 19 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,094評論 25 709
  • 小學(xué)時(shí)?!靶⌒熳?,去給爺爺我買瓶可樂?!蔽以谝巫由下N個(gè)二郎腿說道。“峰…峰哥,我是真…真難得沒錢啦你就別讓我去了。...
    戰(zhàn)毀星辰閱讀 292評論 0 0
  • 故事要有連續(xù)性,有彈性。有一個(gè)連續(xù)的故事主線連著全局。 故事是這樣的:A大學(xué)畢業(yè)慢慢的進(jìn)入社會(huì),學(xué)習(xí)的是廣告的設(shè)計(jì)...
    導(dǎo)演張升志閱讀 281評論 0 0

友情鏈接更多精彩內(nèi)容