asyncio+aiohttp出现Exception ignored:RuntimeError('Event loop is closed')
一、报错信息
1 Exception ignored in: <function _ProactorBasePipeTransport.__del__ at 0x000001B11653EF70> 2 Traceback (most recent call last): 3 File "D:\Codes\SpiderProject\BingImageSpider\demo.py", line 16, in wrappers 4 return func(self, *args, **kwargs) 5 File "D:\DevTools\Python\lib\asyncio\proactor_events.py", line 116, in __del__ 6 self.close() 7 File "D:\DevTools\Python\lib\asyncio\proactor_events.py", line 108, in close 8 self._loop.call_soon(self._call_connection_lost, None) 9 File "D:\DevTools\Python\lib\asyncio\base_events.py", line 746, in call_soon 10 self._check_closed() 11 File "D:\DevTools\Python\lib\asyncio\base_events.py", line 510, in _check_closed 12 raise RuntimeError('Event loop is closed') 13 RuntimeError: Event loop is closed
二、原因分析
像 aiohttp 这类第三方协程库都是依赖于标准库 asyncio 的,而 asyncio 对 Windows 的支持本来就不好。Python3.8 后默认 Windows 系统上的事件循环采用 ProactorEventLoop
(仅用于 Windows )这篇文档描述了其在 Windows 下的缺陷:https://docs.python.org/zh-cn/3/library/asyncio-platforms.html#windows 👈
引发异常的函数是 _ProactorBasePipeTransport.__del__
,所以 aiohttp 铁定使用了 _ProactorBasePipeTransport,并且在程序退出释放内存时自动调用了其__del__
方法
三、解决方案
如果执意要在 Windows 下继续开发,有这几个方案可以选择
1. 不要使用 run 函数
既然 _ProactorBasePipeTransport 会在程序结束后自动关闭事件循环,那就不要用 run 函数了,用官网的例子,使用 loop
loop = asyncio.new_event_loop()
asyncio.set_event_loop(loop)
loop.run_until_complete(main())
2. 替换事件循环
在调用 run 函数前,替换默认的 ProactorEventLoop 为 SelectorEventLoop
asyncio.set_event_loop_policy(asyncio.WindowsSelectorEventLoopPolicy())
asyncio.run(main())
但是 SelectorEventLoop 是有一些缺点的,比如不支持子进程等
3. 忽略异常
这是 Github 上一个外国大佬的方法,在不改变源码的前提下,使用装饰器忽略掉异常
1 import asyncio 2 from asyncio.proactor_events import _ProactorBasePipeTransport 3 from functools import wraps 4 5 import aiohttp 6 7 8 async def main(): 9 async with aiohttp.ClientSession() as session: 10 async with session.get('https://www.baidu.com') as response: 11 print("Status:", response.status) 12 print("Content-type:", response.headers['content-type']) 13 14 html = await response.text() 15 print("Body:", html[:15], "...") 16 17 18 def silence_event_loop_closed(func): 19 @wraps(func) 20 def wrapper(self, *args, **kwargs): 21 try: 22 return func(self, *args, **kwargs) 23 except RuntimeError as e: 24 if str(e) != 'Event loop is closed': 25 raise 26 27 return wrapper 28 29 30 _ProactorBasePipeTransport.__del__ = silence_event_loop_closed(_ProactorBasePipeTransport.__del__) 31 32 asyncio.run(main())
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本