会,Python 默认DNS解析是同步阻塞的,因其底层调用操作系统同步C函数(如getaddrinfo),占用线程并阻塞事件循环;需用aiodns、异步resolver或线程池规避。

会,Python 中默认的 DNS 解析(如 socket.gethostbyname()、socket.getaddrinfo() 或 requests 库发起的 HTTP 请求)是同步阻塞的,会阻塞 asyncio 事件循环。
为什么 DNS 解析会阻塞事件循环?
Python 标准库的 DNS 查询底层调用的是操作系统提供的 C 函数(如 getaddrinfo),这些函数是同步的,执行期间会一直占用当前线程,无法让出控制权给事件循环。即使你在 async def 函数中调用它们,也会导致整个协程暂停,事件循环停摆,其他任务无法推进。
常见触发阻塞的场景
- 直接调用
socket.gethostbyname("example.com") - 使用
httpx.AsyncClient但未配置异步 DNS(默认仍可能走同步解析) - 用
aiohttp发起请求时,若 DNS 缓存未命中且未启用异步 resolver -
requests.get()被误放在async函数中(它本身完全同步)
如何避免阻塞?
核心思路:用真正异步的 DNS 解析方式替代系统默认调用。
-
使用
aiodns:基于 c-ares 的纯异步 DNS 客户端,支持 A、AAAA、MX 等记录查询,不依赖线程池 -
配置 aiohttp / httpx 使用异步 resolver:例如 aiohttp 可传入
aiohttp.AsyncResolver();httpx 支持AsyncHTTPTransport配合自定义 resolver -
用线程池执行同步 DNS(次优解):通过
loop.run_in_executor()将socket.getaddrinfo扔进线程池,避免主线程阻塞,但有上下文切换和资源开销
验证是否真的异步
可通过日志或调试确认 DNS 查询是否发生在事件循环线程内。例如,在 aiodns.DNSResolver.query() 回调中打印 threading.current_thread().name,应为 MainThread;而用 run_in_executor 则会显示 ThreadPoolExecutor 相关线程名。
立即学习“Python免费学习笔记(深入)”;










