python 模塊相互import

模塊A中import B,而在模塊B中import A。這時會怎麼樣呢?這個在Python列表中由RobertChen給出了詳細解釋,抄錄如下:
[A.py]  
from B import D  
class C:pass  

[B.py]  
from A import C  
class D:pass


爲什麼執行A的時候不能加載D呢?

如果將A.py改爲:import B就可以了。

這是怎麼回事呢?

RobertChen:這跟Python內部import的機制是有關的,具體到from B import D,Python內部會分成幾個步驟:

  1. 在sys.modules中查找符號"B"
  2. 果符號B存在,則獲得符號B對應的module對象<module B>。

    從<module B>的__dict__中獲得符號"D"對應的對象,如果"D"不存在,則拋出異常

  3. 如果符號B不存在,則創建一個新的module對象<module B>,注意,這時,module對象的__dict__爲空。

    執行B.py中的表達式,填充<module B>的__dict__ 。

    從<module B>的__dict__中獲得"D"對應的對象,如果"D"不存在,則拋出異常。

所以,這個例子的執行順序如下:

1、執行A.py中的from B import D

由於是執行的python A.py,所以在sys.modules中並沒有<moduleB>存在,首先爲B.py創建一個module對象(<moduleB>),注意,這時創建的這個module對象是空的,裏邊啥也沒有,在Python內部創建了這個module對象之後,就會解析執行B.py,其目的是填充<module B>這個dict。

2、執行B.py中的from A import C

在執行B.py的過程中,會碰到這一句,首先檢查sys.modules這個module緩存中是否已經存在<moduleA>了,由於這時緩存還沒有緩存<moduleA>,所以類似的,Python內部會爲A.py創建一個module對象(<moduleA>),然後,同樣地,執行A.py中的語句。

3、再次執行A.py中的from B import D

這時,由於在第1步時,創建的<moduleB>對象已經緩存在了sys.modules中,所以直接就得到了<moduleB>,但是,注意,從整個過程來看,我們知道,這時<moduleB>還是一個空的對象,裏面啥也沒有,所以從這個module中獲得符號"D"的操作就會拋出異常。如果這裏只是importB,由於"B"這個符號在sys.modules中已經存在,所以是不會拋出異常的。

上面的解釋已經由Zoom.Quiet收錄在啄木鳥了,裏面有圖,可以參考一下:

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章