python - 来自多个模块的全局变量对执行代码不可见?

标签 python module exec

我的应用程序通过 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/

相关文章:

java - 在 Java 中运行 bat 文件并等待

python - 如何将 Python 数据集(之前从 IDL 导出)保存回 IDL 格式

python - 尝试在 Mac 上使用 Python TestRail API - ImportError : No module named Conf_Reader

python - 在 Python 中,单击如何查看其父项具有必需参数的子命令的 --help?

javascript - 从 app.get 回调调用的模块中的主文件访问变量

bash - 找到一个目录并在其中启动可执行文件

python - 如何在基类方法中使用派生类变量

c - 从 tcp 拥塞控制 Linux 内核模块访问 TCP header

javascript - 如何使用默认参数为 Tic-Tac-Toe 游戏中的第一个玩家随机化 "X"和 "O"符号?

c - execvp/fork——如何捕捉不成功的执行?