[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內部會分成幾個步驟:
- 在sys.modules中查找符號"B"
- 果符號B存在,則獲得符號B對應的module對象<module B>。
從<module B>的__dict__中獲得符號"D"對應的對象,如果"D"不存在,則拋出異常
-
如果符號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收錄在啄木鳥了,裏面有圖,可以參考一下: