欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

python解決循環(huán)依賴的問題分析

 更新時間:2022年12月02日 09:30:31   作者:Bruce小鬼  
在使用python開發(fā)過程中在引入其他模塊時可能都經(jīng)歷過一個異常就是循環(huán)引用most likely due to a circular import,它的意思就是A引用了B,反過來B又引用了A,導致出現(xiàn)了循環(huán)引用異常,下面來介紹如何避免循環(huán)引用異常,感興趣的朋友跟隨小編一起看看吧

python解決循環(huán)依賴

1.概述

在使用python開發(fā)過程中在引入其他模塊時可能都經(jīng)歷過一個異常就是循環(huán)引用most likely due to a circular import,它的意思就是A引用了B,反過來B又引用了A,導致出現(xiàn)了循環(huán)引用異常。下面來介紹如何避免循環(huán)引用異常。

2.循環(huán)引用介紹

2.1.python引入模塊原理

下面通過一個循環(huán)引用示例,來介紹python引入模塊的原理。示例中創(chuàng)建了三個模塊,它的引用關系如下

  • dialog.py模塊引入了app模塊的prefs類的get方法
  • app模塊引入了dialog模塊的show方法

創(chuàng)建一個python文件,命名為dialog.py

import app

class Dialog:
    def __init__(self, save_dir):
        self.save_dir = save_dir

save_dialog = Dialog(app.prefs.get('save_dir'))

def show():
    print('Showing the dialog!')

創(chuàng)建一個python文件,命名為app.py

import dialog

class Prefs:
    def get(self, name):
        pass

prefs = Prefs()
dialog.show()

創(chuàng)建一個python文件,命名為main.py

import app

運行上面循環(huán)引用代碼,拋出了異常

AttributeError: partially initialized module 'app' has no attribute 'prefs' (most likely due to a circular import)

要明白上面為什么會拋出循環(huán)引用異常,首先要明白python是如何引入模塊的。在引入模塊的時候,python系統(tǒng)會按照深度優(yōu)先的順序,對模塊執(zhí)行以下五步:

  • 1.在sys.path里尋找模塊的位置
  • 2.把模塊的代碼加載進來,并確認這些代碼能編譯
  • 3.創(chuàng)建響應的空白模塊對象表示該模塊
  • 4.把這個模塊插入sys.modules字典
  • 5.運行模塊對象之中的代碼定義該模塊的內(nèi)容

循環(huán)依賴之所以會出錯,原因在于,執(zhí)行完第4步驟之后,這個模塊已經(jīng)位于sys.modules之中了,然而它的內(nèi)容還沒有得到定義,要等到執(zhí)行完第5步驟,才能齊備。

可是python在執(zhí)行import語句的時候,如果發(fā)現(xiàn)要引用的模塊已經(jīng)出現(xiàn)在了sys.modules之中,(也就是執(zhí)行完第4個步驟),那么就會繼續(xù)執(zhí)行importd 下一條語句,而不會顧及模塊之中的內(nèi)容是否的得到了定義。

例如上面的例子,app模塊在執(zhí)行自己第5步驟時,首先遇到的就是引入dialog模塊的這條語句,而此刻他還沒有把自己的內(nèi)容定義出來,他只不過執(zhí)行完了前4步驟,讓自己出現(xiàn)在了sys.dodules字典里面而已。
等到dialog模塊反過來要引入app的時候,由于app模塊已經(jīng)出現(xiàn)在了sys.modules字典中,python就會認為這個模塊已近引入,于是繼續(xù)執(zhí)行dialog模塊其他代碼,而不會考慮app里面的內(nèi)容到底有沒有定義。
這樣的話,執(zhí)行到save_dialog = Dialog(app.prefs.get('save_dir')) 這一句的時候,就會因為app里面找不到prefs屬性而出錯。(這個屬性必須等app執(zhí)行完第5步驟才能夠得到定義)

3.解決循環(huán)引用方法

如果要解決上面的循環(huán)引用異常,有四種解決辦法。

3.1.重構(gòu)引入關系

例如把prefs內(nèi)容提取到一個單獨的工具模塊中,把它放在依賴體系最底層,這樣app與dialog分別引入這個模塊。他們的關系如下

  • app 引入 prefs
  • dialog 引入 prefs

有時候這種重構(gòu)引入關系需要拆分代碼,對于大型的項目可能不太好拆分,還可以通過其他的方式解決

3.2.調(diào)整import語句

調(diào)整import位置,例如我們可以讓app模塊不要那么早就引入dialog模塊,而是等到prefs等其他內(nèi)容都創(chuàng)建出來之后,在引入dailog,這樣的話,等待dialog返回來使用app中的屬性時,就不會因為該屬性還沒有定義出來而發(fā)生AttributeError

class Prefs:
    def get(self, name):
        pass

prefs = Prefs()

import dialog  # Moved
dialog.show()

這種寫法雖然可行,但是它違背了PEP8規(guī)范,依照建議,所有的import語句都應該出現(xiàn)在文件開頭。這種方式有個弊端,在執(zhí)行了一半,才發(fā)現(xiàn)自己要使用的那個模塊還沒有加載進來,因此不建議使用這種方法。

3.3.把模塊分成引入-配置-運行三個環(huán)節(jié)

循環(huán)引入可以通過勁量縮減引用時所要執(zhí)行的操作。我們可以讓模塊只把函數(shù)、類、與常量定義出來,而不真正去執(zhí)行,這樣python在引入本模塊的時候,就不會由于操作其他模塊而出錯了。
我們可以把本模塊里,需要用到其他模塊的那種操作放在configure函數(shù)中,等到模塊徹底引入完畢后,再去調(diào)用。

dialog.py模塊把調(diào)用的操作放在configure函數(shù)中

import app

class Dialog:
    def __init__(self):
        pass

save_dialog = Dialog()

def show():
    print('Showing the dialog!')

def configure():
    save_dialog.save_dir = app.prefs.get('save_dir')

app.py模塊把調(diào)用的操作放在configure函數(shù)中

import dialog

class Prefs:
    def get(self, name):
        pass

prefs = Prefs()

def configure():
    pass

main.py模塊按照引入-配置-運行的順序先把那兩個模塊引入進來,然后調(diào)用各自的configure函數(shù),最后運行dialog模塊的show函數(shù)

import app
import dialog

app.configure()
dialog.configure()

dialog.show()

這種寫法能適應許多種情況,而且便于我們運用依賴注入模式來替換受依賴模塊之中的內(nèi)容。
但是有時候不太容易從代碼中抽離出這樣一個configure配置環(huán)節(jié),因為他把該模塊定義的對象與這些對象的配置邏輯分別寫到了兩個環(huán)節(jié)里面。

3.4.動態(tài)引入

動態(tài)引入比前幾個方法要簡單,也就是把import語句從模塊級別下移到函數(shù)或方法里面,這樣就解決了循環(huán)依賴關系了。
這種import并不會在程序啟動并初始化本模塊時執(zhí)行,而是等到相關函數(shù)真正運行的時候才得以觸發(fā),因此又叫做動態(tài)引入

下面我們用動態(tài)引入辦法修改dialog模塊,他只會在dialog.show函數(shù)真正運行的時候去引入import模塊,而不像原來那樣,模塊剛初始化,就要引入app

class Dialog:
    def __init__(self):
        pass

# Using this instead will break things
# save_dialog = Dialog(app.prefs.get('save_dir'))
save_dialog = Dialog()

def show():
    import app  # Dynamic import
    save_dialog.save_dir = app.prefs.get('save_dir')
    print('Showing the dialog!')

app模塊修改

import dialog

class Prefs:
    def get(self, name):
        pass

prefs = Prefs()
dialog.show()

main模塊

import 

這樣寫,實際上與剛才那種先引入,再配置,然后運行的辦法是類似的。區(qū)別僅僅在于,這次不調(diào)整代碼的結(jié)構(gòu),也不修改模塊的定義與引入方式,只是把形成循環(huán)依賴的那條import語句推遲到真正需要使用另外一個模塊的那一刻。

一般來說還是勁量避免動態(tài)引入,因為import語句畢竟是有開銷的,如果它出現(xiàn)在需要頻繁執(zhí)行的循環(huán)體里面,那么這種開銷會更大。另外,由于動態(tài)引入會推遲代碼的執(zhí)行時機,有可能你代碼啟動很久之后,如果因為動態(tài)引入其他模塊發(fā)生異常而奔潰。

到此這篇關于python解決循環(huán)依賴的文章就介紹到這了,更多相關python循環(huán)依賴內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • python灰色預測法的具體使用

    python灰色預測法的具體使用

    灰色系統(tǒng)理論認為對既含有已知信息又含有未知或非確定信息的系統(tǒng)進行預測,本文就介紹了python灰色預測法的具體使用,具有一定的參考價值,感興趣的可以了解一下
    2022-03-03
  • 一文總結(jié)學習Python的14張思維導圖

    一文總結(jié)學習Python的14張思維導圖

    一文總結(jié)學習Python的14張思維導圖,本文涵蓋了Python編程的核心知識,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • 詳解PyQt5信號與槽的幾種高級玩法

    詳解PyQt5信號與槽的幾種高級玩法

    這篇文章主要介紹了詳解PyQt5信號與槽的幾種高級玩法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-03-03
  • 淺談Python中的異常和JSON讀寫數(shù)據(jù)的實現(xiàn)

    淺談Python中的異常和JSON讀寫數(shù)據(jù)的實現(xiàn)

    今天小編就為大家分享一篇淺談Python中的異常和JSON讀寫數(shù)據(jù)的實現(xiàn),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-02-02
  • Python matplotlib模塊及柱狀圖用法解析

    Python matplotlib模塊及柱狀圖用法解析

    這篇文章主要介紹了Python matplotlib模塊及柱狀圖用法解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-08-08
  • Python中圖像算術運算的示例詳解

    Python中圖像算術運算的示例詳解

    還記得你在小學時學習如何加減數(shù)字嗎?現(xiàn)在,你也可以對圖像做同樣的事情!輸入圖像可以進行算術運算,例如加法、減法和按位運算(AND、OR、NOT、XOR)。這些操作可以幫助提高輸入照片的質(zhì)量。本文將詳解一下這些運算,需要的可以參考一下
    2022-05-05
  • python分析inkscape路徑數(shù)據(jù)方案簡單介紹

    python分析inkscape路徑數(shù)據(jù)方案簡單介紹

    這篇文章主要介紹了python分析inkscape路徑數(shù)據(jù)方案簡單介紹,文章通過圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的朋友可以參考一下
    2022-09-09
  • django框架模型層功能、組成與用法分析

    django框架模型層功能、組成與用法分析

    這篇文章主要介紹了django框架模型層功能、組成與用法,結(jié)合實例形式簡單分析了Django框架中模型層的基本概念、原理、常用組件構(gòu)成與相關操作技巧,需要的朋友可以參考下
    2019-07-07
  • Python win32com 操作Exce的l簡單方法(必看)

    Python win32com 操作Exce的l簡單方法(必看)

    下面小編就為大家?guī)硪黄狿ython win32com 操作Exce的l簡單方法(必看)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • python+opencv實現(xiàn)文字顏色識別與標定功能

    python+opencv實現(xiàn)文字顏色識別與標定功能

    最近小編接了一個比較簡單的圖像處理的單子,今天小編給大家分享python+opencv實現(xiàn)文字顏色識別與標定功能的完整思路及代碼,感興趣的朋友一起看看吧
    2021-09-09

最新評論