我有两个模块如下:
模块 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(((是None。我认为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')
。moda.py
中定义的__main__
完成填充其命名空间(从步骤 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
.