微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

在协程中尝试/除外/最终

如何解决在协程中尝试/除外/最终

class DemoException(Exception):   
    """An exception type for the demonstration."""

def demo_exc_handling():
    print('-> coroutine started')
    while True:
        try:
            x = yield
        except DemoException:  # <1>
            print('*** DemoException handled. Continuing...')
        else:  # <2>
            print('-> coroutine received: {!r}'.format(x))
        finally:
            print('-> 1111111111coroutine ending')
    raise RuntimeError('This line should never run.')  

if __name__ == '__main__':
exc_coro = demo_exc_handling()
next(exc_coro)
exc_coro.send(11)

我得到以下输出

-> coroutine started
-> coroutine received: 11
-> 1111111111coroutine ending
-> 1111111111coroutine ending

我想知道为什么finally语句执行两次? 我将非常感谢您的帮助。

解决方法

这与协程或异常无关。只需生成器即可重现:

def demo_exc_handling():
    print('-> coroutine started')
    while True:
        try:
            x = yield
        finally:
            print('-> 1111111111coroutine ending')
    raise RuntimeError('This line should never run.')

if __name__ == '__main__':
    exc_coro = demo_exc_handling()
    next(exc_coro)
    next(exc_coro)

以下是输出:

-> coroutine started
-> 1111111111coroutine ending
-> 1111111111coroutine ending

您可能想做的是在try-finally内部进行while循环:

def demo_exc_handling():
    print('-> coroutine started')
    try:
        while True:
            x = yield
    finally:
        print('-> 1111111111coroutine ending')
    raise RuntimeError('This line should never run.')

而是输出

-> coroutine started
-> 1111111111coroutine ending
,

它打印两次,因为即使主程序结束,finally子句也会执行。

协程在主程序结束时第二次产生

让我们产生迭代次数并添加一些打印输出以查看它

class DemoException(Exception):   
    """An exception type for the demonstration."""
    pass

def demo_exc_handling():
    print('-> coroutine started')
    i = 1
    while True:
        try:
            print(f"iteration {i}")
            x = yield i
        except DemoException:  # <1>
            print('*** DemoException handled. Continuing...')
        else:  # <2>
            print('-> coroutine received: {!r}'.format(x))
        finally:
            print('-> 1111111111coroutine ending')
        i += 1
    raise RuntimeError('This line should never run.')  

if __name__ == '__main__':
    exc_coro = demo_exc_handling()
    print("initialised")
    y = next(exc_coro)
    print(f"got {y}")
    exc_coro.send(11)

产生

initialised
-> coroutine started
iteration 1
got 1
-> coroutine received: 11
-> 1111111111coroutine ending
iteration 2
-> 1111111111coroutine ending

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?