我的应用程序通过 exec 执行存储在配置文件中的 python 逻辑,如下所示:
"foo() + 2"
此逻辑通常引用我存储在名为“standard”的模块中的符号。例如,在上面,您可以看到逻辑访问方法 foo(),并且 foo 定义在 standard.py 中:
def foo():...
为了向逻辑提供对标准中符号的访问,我将标准中的方法提取到字典中,如下所示:
import standard
my_globals = standard.__dict__
然后,我向 my_globals 添加一些其他相关符号(此处未显示),并在执行时将它们提供给逻辑:
exec("foo() + 2", my_globals)
这正在工作。当我从 foo() 内部查看 globals() 时,我可以看到我在模块 standard.py 中定义的其他方法以及我上面提到的其他相关符号,并且 foo() 可以访问所有这些东西。
当我想让另一个功能模块也可供逻辑使用时,问题就出现了。假设我有一个名为 custom.py 的模块,其中有一些我希望逻辑访问的其他符号。我正在尝试通过这样做来使这些符号可用:
import custom
my_globals.update(custom.__dict__)
假设我现在的逻辑是“bar() + 1”,其中“bar”是在 custom.py 内部定义的。 bar() 还想访问我添加到 my_globals 中的一些相关其他符号。
我遇到的问题是,custom 内部的代码只能看到 custom.py 中定义的符号,而不是 my_globals 中存储的其他所有内容。 IE 中,bar() 看不到 foo(),也看不到我藏在 my_globals 中的其他东西。
但是 foo() 可以。它的代码可以看到我在标准中定义的任何其他方法,以及在自定义中定义的符号,以及我插入 my_globals 中的“额外”符号。
为什么会发生这种情况?我的期望是正在执行的逻辑是在 my_globals 内容的上下文中运行的,因此 foo() 和 bar() 似乎都应该能够访问 my_globals 中的任何和所有符号。
我怀疑这与我创建 my_globals 的方式有关。任何见解将不胜感激!
最佳答案
这里有一些见解:
"To provide the logic with access to the symbols in standard, I'm extracting out the methods from standard into a dictionary, like so:"
import standard
my_globals = standard.__dict__
不完全是。您只需创建一个局部变量 my_globals
,它现在指向 standard.__dict__
。每当您更新 my_globals
时,您实际上只是更新 standard.__dict__
。
当您将其他符号添加到 my_globals
时,您只是将它们添加到 standard.__dict__
。
调用:
exec("foo() + 2", my_globals)
在 standard.py 中定义 foo
时效果很好,因为您已将所有其他方法添加到此模块中 - 现在您可以访问所有这些方法。
当你这样做时:
import custom
my_globals.update(custom.__dict__)
您已将 custom.py 中的“符号”添加到标准模块中。此后,标准中的所有函数都可以访问 custom.py 中的函数
不幸的是,custom.py 本身无法直接访问 standard.py 中的方法(除非您导入它们)。从 custom.py 中,您可以看到您创建的所有内容现在都是标准的:
(来自custom.py):
import sys
def custom_foo():
print(dir(sys.modules['standard'])) # shows that you've put everything in here
sys.modules['standard'].foo() # calls foo from standard.py (assuming you've imported standard in your main pgm)
上面的内容确实很难看 - 你可以添加一个:
from standard import *
位于 custom.py 的顶部,您将可以访问添加到其 __dict__
中的所有内容。
我怀疑您真的想要对整个exec
事情做您正在尝试的事情,但我不太确定您的用例是什么。
编辑:
如果您确实希望附加到 my_globals
的所有符号可用于 custom.py 的方法,您可以调用:
custom.__dict__.update(my_globals)
此后,custom.py 中的函数将可以访问您添加到标准中的所有内容。dict(又名 my_globals)。 (您还使用 my_globals
中的同名函数覆盖了 custom.py 中定义的任何函数)
请注意,以这种方式做事是非常不典型的(阅读:有点不明智)。
关于python - 来自多个模块的全局变量对执行代码不可见?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33268695/