为什么我不能在python中更改另一个模块变量?

Bil*_*man 4 python variables import circular-dependency

我试图允许第二个模块在循环导入中修改第一个模块的变量,但它似乎不起作用.

我有两个问题:1)为什么这不起作用/从语言发展的角度来看这是什么原因2)是否有任何简单的解决方案可以让我以略微不同的方式做同样的事情?

a.py:

import b

test1 = 'a'
test2 = None
test3 = '3'

if __name__ == '__main__':
  print test1, test2, test3 #prints 'a', None, 3
  b.changeVars()
  print test1, test2, test3 #prints 'a', None, 3 (i.e. nothing has changed)
Run Code Online (Sandbox Code Playgroud)

b.py:

import a

def changeVars():
  print a.test1, a.test2, a.test3 #prints 'a', None, 3
  a.test1 = 'NEW VAR 1'
  a.test2 = 'NEW VAR 2'
  a.test3 = 'NEW VAR 3'

  print a.test1, a.test2, a.test3 #prints 'NEW VAR 1', 'NEW VAR 2', 'NEW VAR 3'
Run Code Online (Sandbox Code Playgroud)

aar*_*ing 8

发生的事情是,当b.py尝试时import a,sys.modules由于条目不足,因此没有条目__main__.这会导致导入机制重新导入模块并将其置于名称下a.所以现在有一个a模块和一个完全不相关的__main__模块.换成b.py这样的东西就可以了.

import sys
a = sys.modules['__main__']

def changeVars():
  print a.test1, a.test2, a.test3 #prints 'a', None, 3
  a.test1 = 'NEW VAR 1'
  a.test2 = 'NEW VAR 2'
  a.test3 = 'NEW VAR 3'

  print a.test1, a.test2, a.test3 #prints 'NEW VAR 1', 'NEW VAR 2', 'NEW VAR 3'
Run Code Online (Sandbox Code Playgroud)

产量

aaron@aaron-laptop:~/code/tmp$ python a.py
a None 3
a None 3
NEW VAR 1 NEW VAR 2 NEW VAR 3
NEW VAR 1 NEW VAR 2 NEW VAR 3
aaron@aaron-laptop:~/code/tmp$ 
Run Code Online (Sandbox Code Playgroud)

要更好地了解正在发生的事情,请考虑以下文件:

#a.py
import b
import a

test = 'Foo'   

if __name__ == '__main__':
  print test  #prints 'Foo'
  b.changeVars()
  print a.test, test # prints 'Foo', 'Bar'
Run Code Online (Sandbox Code Playgroud)

#b.py
import a as a1

import sys
a2 = sys.modules['__main__']

def changeVars():

  print a1.test, a2.test # Prints 'Foo', 'Foo'
  a2.test = 'Bar'
  print a1.test, a2.test # Prints 'Foo', 'Bar'
Run Code Online (Sandbox Code Playgroud)

哪个输出

Foo
Foo Foo
Foo Bar
Foo Bar
Run Code Online (Sandbox Code Playgroud)

这清楚地表明,sys.modules['a']sys.modules['__main__']指的是两个不同的对象.解决方案可能是将以下内容作为第一行a.py

import __main__ as a  # due to Ignacio Vazquez-Abrams 
Run Code Online (Sandbox Code Playgroud)

这样做也允许任何其他模块import a.总的来说,我真的不明白你为什么要这样做.完成这项工作可能有更好的方法.

  • `import __main__ as a` 但是,是的,不要这样做。 (2认同)