显式关闭asyncio事件循环的必要性

时间:2016-12-19 21:34:37

标签: python python-3.x python-asyncio

故事:

我目前正在查看asyncio basic examples,特别是this one - 最简单的HTTP客户端。 main函数启动一个事件循环,一直运行直到数据提取完成并关闭事件循环:

def main():
    loop = get_event_loop()
    try:
        body = loop.run_until_complete(fetch())
    finally:
        loop.close()
    print(body.decode('latin-1'), end='')

但是,如果省略loop.close()

,代码也可以
def main():
    loop = get_event_loop()
    body = loop.run_until_complete(fetch())
    print(body.decode('latin-1'), end='')

问题:

虽然有一个例子,但问题是通用问题 - 如果忘记关闭asyncio事件循环,可能会出现什么问题?事件循环是否总是隐式关闭?

1 个答案:

答案 0 :(得分:5)

不同的事件循环实现可以使用

.close()来释放循环分配的系统资源(或执行任何其他操作)。如果您要查看_UnixSelectorEventLoop的代码,这是Linux中使用的(默认)IOLoop,您将找到以下代码:

def close(self):
    super().close()
    for sig in list(self._signal_handlers):
        self.remove_signal_handler(sig)

例如,close()删除了使用loop.add_signal_handler()注册的信号处理程序。

由于可以在不同的线程上启动多个IOLoops,或者可以在旧的IOLoops关闭之后创建新的IOLoops(请参阅asyncio.new_event_loop()),因此关闭它们应该被认为是一个好习惯。

<强>更新

从Python 3.7开始,建议使用asyncio.run而不是run_until_complete()

# Python 3.7+
def main():
    body = asyncio.run(fetch())
    print(body.decode('latin-1'), end='')

除其他事项外,asyncio.run负责finally close()循环。