- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在阅读 python 文档以改进我的核心 python,我正在阅读有关 errors and exceptions 的内容
在文档中说
If a finally clause includes a return statement, the finally clause’s return statement will execute before, and instead of, the return statement in a try clause.
下面还提供了这个例子:
def bool_return():
try:
return True
finally:
return False
bool_return()
现在看这个例子,上面的陈述看起来很直接和公平,但是如果你稍微修改那个例子让它看起来像这样:
def bool_return():
try:
return print("foo")
finally:
return False
bool_return()
现在,如果你运行它,你会看到 foo
将被打印并返回 False。现在文档说 finally 子句的返回将执行 before,而不是 try 子句的 return 语句。如果是这样,那为什么我可以看到正在打印的 foo?
我用 pycharm 调试了这段代码,它显示首先执行 try 子句的 return 语句并打印字符串,然后输出 None
由于 return
被退回语句,finally 子句中的 return 语句将在后面执行,这是程序的最后一次返回,因此函数覆盖了之前的 return 和 False
被退回。
我的问题是:
1) 为什么文档说 finally 子句的 return 语句在 之前执行?
2) 为什么 doc 说执行 finally 子句的 return 语句而不是try 子句的 return 语句?
我相信这两种说法都与现实情况相反。
编辑:
阅读@iBug 的回答后很清楚 print("foo")
是如何实现的被评估但是None
不返回。基本上,首先计算表达式,然后计算 return
。发生。稍后return False
in finally 被执行。这清楚地说明了为什么我们得到了我们所做的输出。
我仍然看到 return False
在 return print("foo")
之后 终于执行了的尝试。
或者根据@iBug 的评论,10 RETURN_VALUE
被完全绕过?
编辑
这个问题现在在文档中得到了解决,现在返回的内容是正确的。但是,如果您想知道“如何”,请阅读所有评论并仔细回答。
最佳答案
$ python3
Python 3.7.5 (default, Nov 20 2019, 09:21:52)
[GCC 9.2.1 20191008] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> def bool_return():
... try:
... return print("foo")
... finally:
... return False
...
>>> import dis
>>> dis.dis(bool_return)
2 0 SETUP_FINALLY 8 (to 10)
3 2 LOAD_GLOBAL 0 (print)
4 LOAD_CONST 1 ('foo')
6 CALL_FUNCTION 1
8 RETURN_VALUE
5 >> 10 LOAD_CONST 2 (False)
12 RETURN_VALUE
>>>
正如您在上面看到的,return False
确实发生在 try
block 中的 return
语句之前,但是在计算出要返回的值之后。
我认为文档可能意味着 return 语句 的“返回的 Action ”,或者换句话说,它没有考虑返回值的计算,这当然会发生在返回之前。
要观察8 RETURN_VALUE
是否执行,可以在 Debug模式下编译CPython解释器,然后在GDB中运行。分步指南对于这个答案来说过于臃肿,所以我将在此处给出一个大纲 (Linux)。
./configure --with-pydebug
(您可能还想提供 --prefix=/opt/python3-debug
),make
和 make install
gdb/opt/python3-debug/bin/python3
和 (gdb) r
bool_return
。Python/ceval.c
中找到字符串 RETURN_VALUE
,记下行号 (for 3.8.1, it's 1911)。SIGTRAP
挂起Python解释器,并在上一步的位置设置断点(b Python/ceval.c:1911
),然后c
。(gdb breakpoint info)
False
(gdb breakpoint info)
return
语句。既然清楚了函数中只执行了一次return
,那肯定是12 RETURN_VALUE
,所以Python指令8 RETURN_VALUE
根本不执行。
关于Python 文档对 finally 中的 return 有误导性的解释,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59639733/
关闭。这个问题需要更多 focused .它目前不接受答案。 想要改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭 2 年前。 Improve this q
1.final final修饰类,说明这个类不能被继承,是以个顶级类。 final修饰变量,说明这个变量是常量。 final修饰方法,表示这个方法不能被重写,不过可以冲在final方法。 比如有个
我喜欢使用 -Wsuggest-final-types 编译我的代码和 -Wsuggest-final-methods以便在可能使用 final 关键字以允许编译器更积极地优化的机会时收到警告。 不过
我在 Java 8 中有一个异步操作,它返回一个 onError 回调或一个 onSuccess 回调。如果操作成功与否,我需要在我的方法内部返回。所以我返回一个 boolean 值来说明这个信息。我
我正在阅读一些内容,需要对最终类和方法进行一些说明。我的理解是,将一个类声明为 final 会阻止该类被扩展。因此,是否有必要将最终类中的方法声明为最终的?在我看来,如果类不能扩展,则没有必要将方法声
有什么区别 try { // action A } catch(Exception e) { // action B } finally { // action C }
这个程序是我类(class)的最终作业,我无法弄清楚为什么我收到错误“从内部类引用的局部变量必须是最终的或有效的最终”。该程序正在运行并发线程来对# 数组进行排序,然后找到该数组的高值和低值。当我在没
C++11 added final. 终于! 我了解 final 做了两件事: 使类不可继承。 使类中的(虚拟)函数不可覆盖(在派生类中)。 这两者似乎是相互独立的。但以以下为例: class Foo
我想使用具有多个提交按钮的react-final-form构建表单,其中每个提交按钮在表单中设置不同的值。本质上,我想创建一个在呈现的HTML中看起来像这样的表单: Are you over 1
我想知道什么时候应该对变量和(或)方法使用静态、最终、静态最终参数。据我了解: final:类似于c++中的const参数。它基本上意味着值(或在方法中 - 返回值)不会改变。 静态:这意味着值(或方
我正在做一个编程类(class)项目,用于 400 行矩阵本身的矩阵乘法。我让它以顺序模式工作,该项目的目标是编写并行实现。 我有以下代码,当然,当我尝试引用内部类中的计数器 j 时,我收到一个关于
也许这是简单的问题,但我找不到答案。 声明为final的经典变量是否包含与非final变量不同的内存段? 最佳答案 我想说,谈到局部变量,基于 this answer 是不行的。 : The trut
考虑以下代码: #include class B { virtual void f() { std::cout << "Base" << '\n'; } }; class D
这个问题已经有答案了: java: Is it possible to set a lambda expression for an array of Buttons is a for loop? I
考虑这个代码片段 public class ConstantFolding { static final int number1 = 5; static final int numbe
我确定在内部类中访问的变量应该声明为final 或有效final。但在下面的情况下我很困惑。不知道是不是我理解错了。 public class MyClass { private boolea
我必须将一个 java.sql.Connection 对象传递给一个匿名内部类,这意味着我必须对它进行 final 引用。但是,我担心任何资源泄漏。 public static String foo(
我收到 SonarQube 错误:“强烈建议在此方法实现结束时调用 super.finalize(),以防父实现也必须释放一些系统资源。” 但我发现 Object 类没有实现 finalize方法。
我一般认为资源清理是在 finally block 中完成的, 最近我在一个类中发现了这个特定的代码片段,它覆盖了 Object 类'finalize()方法。 protected void fina
让我们在父类中使用这个方法: public void calculateSum(int a, final int b) { } 子类有: public void calculateSum(int a,
我是一名优秀的程序员,十分优秀!