- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
首先,我要说的是,我阅读了很多关于创建动态命名变量的类似主题的帖子,但它们大多与 Python 2 相关,或者它们假定您正在使用类。是的,我读了Behavior of exec function in Python 2 and Python 3 .
我也知道创建动态命名的变量在 99% 的时间里是一个坏主意,字典是获得的方式,但我只想知道它是否仍然可能以及 exec 和 locals 在 python 中的工作方式3.
我想展示一些示例代码来说明我的问题(fibonacci 计算斐波那契数,ListOfLetters 提供 ["A", "B", ...]):
def functionname():
for index, buchstabe in enumerate(ListOfLetters.create_list("A", "K"), 1):
exec("{} = {}".format(buchstabe, fibonacci(index)) ) #A = 1, B = 1, C = 2, D = 3, E = 5,...
print(index, buchstabe, eval(buchstabe)) #works nicely, e.g. prints "4 D 3"
print(locals()) #pritns all locals: {'B': 1, 'A': 1, 'index': 11, 'C': 2, 'H': 21, 'K': 89, ...
print(locals()['K']) #prints 89 as it should
print(eval("K")) #prints 89 as it should
print(K) #NameError: name 'K' is not defined
所以至少在我目前的理解中,locals()
的行为存在一些不一致,因为它包含由 exec()
添加的变量名,但是变量在函数中不可用。
如果有人可以对此进行解释并说明这是设计使然还是语言中的真正不一致,我将不胜感激。是的,我知道 locals
不应该被修改,但我没有修改它,我正在调用 exec()
...
最佳答案
当您不确定为什么某些东西会以它在 Python 中的方式工作时,通常可以将您感到困惑的行为放在一个函数中,然后使用 dis 从 Python 字节码中反汇编它
模块。
让我们从更简单的代码开始:
def foo():
exec("K = 89")
print(K)
如果您运行 foo()
,您将得到与您在更复杂的函数中看到的相同的异常:
>>> foo()
Traceback (most recent call last):
File "<pyshell#167>", line 1, in <module>
foo()
File "<pyshell#166>", line 3, in foo
print(K)
NameError: name 'K' is not defined
让我们拆开它看看为什么:
>>> import dis
>>> dis.dis(foo)
2 0 LOAD_GLOBAL 0 (exec)
3 LOAD_CONST 1 ('K = 89')
6 CALL_FUNCTION 1 (1 positional, 0 keyword pair)
9 POP_TOP
3 10 LOAD_GLOBAL 1 (print)
13 LOAD_GLOBAL 2 (K)
16 CALL_FUNCTION 1 (1 positional, 0 keyword pair)
19 POP_TOP
20 LOAD_CONST 0 (None)
23 RETURN_VALUE
您需要注意的操作是标记为“13”的操作。这是编译器在函数的最后一行 (print(K)
) 中处理查找 K
的地方。它使用 LOAD_GLOBAL
操作码,它失败了,因为“K”不是全局变量名,而是我们的 locals()
字典中的一个值(由 exec
调用)。
如果我们说服编译器将 K
视为局部变量(通过在运行 exec
之前给它一个值),它会知道不要寻找一个不存在的全局变量?
def bar():
K = None
exec("K = 89")
print(K)
如果你运行这个函数,它不会给你一个错误,但是你不会打印出期望值:
>>> bar()
None
让我们反汇编看看为什么:
>>> dis.dis(bar)
2 0 LOAD_CONST 0 (None)
3 STORE_FAST 0 (K)
3 6 LOAD_GLOBAL 0 (exec)
9 LOAD_CONST 1 ('K = 89')
12 CALL_FUNCTION 1 (1 positional, 0 keyword pair)
15 POP_TOP
4 16 LOAD_GLOBAL 1 (print)
19 LOAD_FAST 0 (K)
22 CALL_FUNCTION 1 (1 positional, 0 keyword pair)
25 POP_TOP
26 LOAD_CONST 0 (None)
29 RETURN_VALUE
注意“3”和“19”处使用的操作码。 Python 编译器使用 STORE_FAST
和 LOAD_FAST
将局部变量 K
的值放入槽 0,然后再将其取回。使用带编号的插槽比从 locals()
这样的字典中插入和获取值要快得多,这就是为什么 Python 编译器对函数中的所有局部变量访问都这样做。您不能通过修改 locals()
返回的字典来覆盖插槽中的局部变量(如 exec
所做的那样,如果您不向它传递要使用的字典为其命名空间)。
确实,让我们尝试我们函数的第三个版本,当我们将 K
定义为常规局部变量时,我们再次查看 locals
:
def baz():
K = None
exec("K = 89")
print(locals())
这次您也不会在输出中看到 89
!
>>> baz()
{"K": None}
您在 locals()
中看到旧的 K
值的原因在 the function's documentation 中有解释。 :
Update and return a dictionary representing the current local symbol table.
局部变量K
的值所在的槽没有被exec
语句改变,它只修改了locals()
字典。当您再次调用 locals()
时,Python 使用槽中的值“更新[s]”字典,用 exec
替换存储在那里的值。
这就是文档继续说的原因:
Note: The contents of this dictionary should not be modified; changes may not affect the values of local and free variables used by the interpreter.
您的 exec
调用正在修改 locals()
字典,您发现它的更改并不总是被您以后的代码看到。
关于python - 在 python 3 中的函数中创建动态命名变量/理解 python 3 中的 exec/eval/locals,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25076883/
我正在尝试设置 rbenv: https://github.com/sstephenson/rbenv#section_2 我有一个问题:我必须在一切之前使用 rbenv exec 所以现在我要做 b
我已经使用我们的应用程序创建了图像,运行该图像后,我可以看到还创建了 docker 容器,当我尝试进入 docker 容器时,出现以下错误,请您在这里帮助我。 """"OCI runtime exec
这post on javapapers.com显示如何运行 JMH通过键入 mvn exec:exec 在 Maven 中进行基准测试。在 Maven 中运行 JMH 非常方便,因为您可以轻松地从 E
因为我是 shell 脚本的新手,exec命令总是让我感到困惑,同时用 while 探索这个话题循环触发了以下 4 个问题: 下面的语法 1 和 2 有什么区别 语法 1: while read LI
湖一看这两条线 案例 1 set cvsUpdStr [exec cvs -qn upd] 案例 2 set cvsUpdStr [exec cvs -qn upd >&@stdout] 对于第一种情
我有一个配置为使用 Maven 构建和运行的项目。该项目取决于平台特定的 native 库,我正在使用发现的策略 here管理这些依赖项。 本质上,特定平台的 .dll 或 .so 文件被打包到 ja
我发现 tcl exec 命令首先从标准输出返回字符串,然后是标准错误。例如,我下面的“测试脚本”按以下顺序生成消息: puts "test started" puts stderr "some no
我有 rpc error: code = 13 desc = invalid header field value "oci runtime error: exec failed: cannot ex
过去 2 天我遇到了一个问题。 我正在运行一个 tcl 脚本(用于 eggdrop),该脚本在被触发时执行本地 shell 命令(子进程),如果命令成功,它会输出结果。但是,如果命令不成功,我会收到错
假设我有一个像这样的 TCL 脚本: exec ls -l 现在这将打印出当前目录的内容。我需要将该输出作为字符串并解析它。我怎么能做到这一点? 最佳答案 exec返回输出,因此只需为其设置一个变量:
我正在尝试在 PHP 中使用 GNUPLOT 绘制正弦图,但是当我使用 exec 绘制该图时出现此错误: Warning: exec() [function.exec]: Cannot execute
我正在尝试 mvn exec:exec (或 mvn exec:java )使用类路径中的本地 jar 运行我的程序。但是 jar 无法加载: Exception in thread "main" j
我在最新的 Docker CLI 文档中注意到 Docker CLI 命令列表已经扩展。 如果我使用 码头执行 早些时候在容器内启动可执行文件现在我也可以使用 docker 容器执行 命令。 dock
这个问题在这里已经有了答案: Call to a member function exec() on a non-object error in PHP [duplicate] (1 个回答) 关闭
手册页说“exec() 系列函数用新的过程镜像替换当前的过程镜像。”但我不太明白“用新过程图像替换当前过程图像”的含义。比如exec成功,perror就达不到 execl("/bin/ls", /*
手册页说“exec() 系列函数用新的过程镜像替换当前的过程镜像。”但我不太明白“用新过程图像替换当前过程图像”的含义。比如exec成功,perror就达不到 execl("/bin/ls", /*
我正在 UrbanCode Deploy Shell 步骤中执行以下命令。我想要的是针对找到的所有文件运行 Util.sh。 find . -type f -exec ls -al {} \; fin
仅当将“-p”添加到 org.codehaus.mojo:exec-maven-plugin:1.6.0:exec 插件时,以下 pom.xml 执行才会失败,删除此参数后它运行正常,但我需要争论,有
我正在尝试进入 Maven 并开始编写一个小型库。作为库代码的一部分,我想要一个小型演示 Java 应用程序,可以用来查看库在做什么以及如何使用它。现在真正让我烦恼的一件事是,为了执行该演示,我必须执
我是 maven 的新手,在通过 maven 运行类文件时遇到问题 它运行良好 mvn exec:java -Dexec.mainClass="com.test.Test" 但不是 mvn exec:
我是一名优秀的程序员,十分优秀!