我有两个模块,如下所示:

模块A-moda.py

import modb

x = None

def printx():
    global x
    print(x)

def main():
    global x
    x = 42
    printx()
    modb.printx()
    printx()

if __name__ == '__main__':
    main()

模块B-modb.py

import moda

def printx():
    moda.printx()

print('modb imported')

当我运行python moda.py时,我得到的输出是:
modb imported
42
None
42

我不明白为什么第二个打印(来自modb.printx())是“无”。我认为python模块表现为单例。我想念什么?

有人可以解释为什么modb中导入的模块与原始模块moda不同吗?

最佳答案

当遇到import语句时,解释器会在sys.modules中查找相应的键。如果找到密钥,则该密钥将绑定到您请求的名称。如果不是,则在sys.modules中创建一个新的空模块对象,然后进行填充。这样做的原因恰恰是为了避免循环引入的无限循环。

运行模块时,将以__main__的名称导入该模块。你约

这是当您将moda作为脚本运行时的事件序列:

  • 开始加载moda.py作为sys.modules['__main__']。至此,这只是一个空的命名空间
  • import modb中遇到的
  • moda.py。为sys.modules['modb']创建的新的空名称空间。
  • import moda中遇到的
  • modb.py。为sys.modules['moda']创建的新的空名称空间。请注意,这是,而不是,与步骤1中的sys.modules['__main__']是同一对象。
  • import modb中遇到的
  • moda.py。由于sys.modules['modb']存在,因此将其绑定到moda中的该名称
  • 由于当前正在以moda.py的名称加载moda,因此无需运行导入保护即可完成其名称空间的填充。
  • modb.py完成填充其名称空间(从步骤2开始)并运行print('modb loaded')
  • __main__中定义的
  • moda.py完成填充其名称空间(从步骤1开始)并运行导入保护。

  • 希望这可以帮助您可视化发生的情况。您已加载了三个模块,而不是两个模块,因为moda是用两个不同的名称加载的,并且是作为两个完全不同的模块对象加载的。
    __main__中的导入保护调用__main__.main,它执行以下操作:
  • 设置__main__.x = 42(moda.x仍是None)
  • __main__.printx显示__main__.x,即42
  • modb.printx调用moda.printx,它打印moda.x,即None
  • __main__.printx再次打印__main__.x,仍然是42
  • 10-04 15:08