故事:
我目前正在查看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事件循环,可能会出现什么问题?事件循环是否总是隐式关闭?
答案 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()
循环。