Python协程全解析:手把手教你用asyncio撸协程
时间:2025-06-22 14:38:26 198浏览 收藏
golang学习网今天将给大家带来《Python协程深度解读,手把手教你用asyncio实现协程》,感兴趣的朋友请继续看下去吧!以下内容将会涉及到等等知识点,如果你是正在学习文章或者已经是大佬级别了,都非常欢迎也希望大家都能给我建议评论哈~希望能帮助到大家!
协程是一种比线程更轻量级的并发方式,基于生成器或async/await语法实现,通过主动让出CPU控制权提升程序效率。1.Asyncio是Python中基于协程的并发库,使用事件循环调度任务;2.协程由程序自身调度,资源消耗小,避免了锁和死锁问题;3.避免阻塞操作需使用异步I/O库如aiohttp或放到独立线程执行;4.异常处理可通过try...except捕获,或使用asyncio.gather配合return_exceptions=True参数;5.调试可通过启用调试模式、添加日志、使用调试器或第三方库进行。示例代码展示了协程创建、并发执行及异常处理机制。
协程,简单来说,就是一种比线程更轻量级的并发方式。它允许你在单线程中执行多个任务,通过主动让出CPU控制权来实现任务间的切换,从而提高程序的效率。Asyncio是Python中用于编写并发代码的库,它基于协程实现。

Python协程的运作方式和基本原理
Python的协程基于生成器(generator)或async/await语法实现。生成器方式比较老旧,现在更推荐使用async/await。Async/await实际上是语法糖,底层还是基于生成器实现的。

async关键字用于声明一个协程函数,await关键字用于挂起协程的执行,等待另一个协程完成。当一个协程遇到await时,它会主动让出CPU控制权,允许事件循环(event loop)调度其他协程执行。
这种方式避免了线程切换的开销,因为协程的切换是由程序自身控制的,而不是由操作系统内核控制的。因此,协程的效率通常比线程更高。

import asyncio async def my_coroutine(delay): print(f"Coroutine started, sleeping for {delay} seconds...") await asyncio.sleep(delay) print("Coroutine finished!") async def main(): task1 = asyncio.create_task(my_coroutine(2)) task2 = asyncio.create_task(my_coroutine(1)) await asyncio.gather(task1, task2) # 等待所有任务完成 if __name__ == "__main__": asyncio.run(main())
在这个例子中,my_coroutine
是一个协程函数,它会休眠一段时间然后打印消息。main
函数创建了两个 my_coroutine
的任务,并使用 asyncio.gather
同时运行它们。注意,asyncio.sleep
也是一个协程,它会挂起当前协程的执行,允许事件循环调度其他协程。
协程与多线程的区别?
协程和多线程都是并发编程的方式,但它们有着本质的区别。
- 调度方式: 线程由操作系统内核调度,属于抢占式调度。协程由程序自身调度,属于协作式调度。
- 资源消耗: 线程的创建和切换开销较大,需要占用较多的系统资源。协程的创建和切换开销很小,只需要占用较少的内存。
- 同步方式: 多线程需要使用锁等同步机制来避免竞争条件。协程由于在单线程中执行,不需要使用锁,避免了死锁等问题。
简单来说,你可以把线程想象成多个独立的程序,而协程则是同一个程序中的多个子程序。协程更轻量级,更易于管理,但也需要开发者自己负责任务的调度。
如何避免Asyncio协程中的阻塞操作?
Asyncio 的核心在于非阻塞 I/O。如果你的协程中包含了阻塞操作(比如同步的网络请求、文件读写等),那么整个事件循环都会被阻塞,导致其他协程无法执行。
避免阻塞操作的关键在于使用异步的 I/O 库。例如,可以使用 aiohttp
替代 requests
进行异步的 HTTP 请求,使用 aiofiles
替代内置的 open
函数进行异步的文件读写。
如果必须执行阻塞操作,可以使用 asyncio.to_thread
将其放到一个独立的线程池中执行,从而避免阻塞事件循环。
import asyncio import aiohttp async def fetch_url(url): async with aiohttp.ClientSession() as session: async with session.get(url) as response: return await response.text() async def main(): url = "https://www.example.com" content = await fetch_url(url) print(f"Fetched content from {url}: {content[:100]}...") if __name__ == "__main__": asyncio.run(main())
在这个例子中,aiohttp
库用于执行异步的 HTTP 请求,避免了阻塞事件循环。
如何处理Asyncio协程中的异常?
在协程中处理异常的方式与普通函数类似,可以使用 try...except
语句捕获异常。但是,需要注意的是,如果一个协程中发生了未捕获的异常,它可能会导致整个事件循环崩溃。
为了避免这种情况,可以使用 asyncio.create_task
创建任务时,显式地处理异常。
import asyncio async def my_coroutine(): await asyncio.sleep(1) raise ValueError("Something went wrong!") async def main(): task = asyncio.create_task(my_coroutine()) try: await task except ValueError as e: print(f"Caught an exception: {e}") if __name__ == "__main__": asyncio.run(main())
另外,asyncio.gather
也可以用于处理异常。如果传递给 asyncio.gather
的任何一个协程抛出异常,asyncio.gather
也会抛出同样的异常。你可以使用 return_exceptions=True
参数来让 asyncio.gather
返回所有协程的结果,包括异常。
如何调试Asyncio协程?
调试 asyncio 协程可能会比较棘手,因为代码的执行流程不是线性的。
- 启用调试模式: 可以通过设置
PYTHONASYNCIODEBUG=1
环境变量来启用 asyncio 的调试模式。这会输出更多的调试信息,帮助你找到问题所在。 - 使用日志: 在协程中添加日志语句,可以帮助你了解代码的执行流程和状态。
- 使用调试器: 可以使用 Python 的调试器(如 pdb)来调试协程。但是,需要注意的是,调试器可能会干扰事件循环的执行,导致一些奇怪的问题。
- 使用第三方库: 有一些第三方库可以帮助你调试 asyncio 协程,例如
aiodebug
。
总之,调试 asyncio 协程需要耐心和技巧。多尝试不同的方法,总能找到问题所在。
今天带大家了解了的相关知识,希望对你有所帮助;关于文章的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
419 收藏
-
482 收藏
-
422 收藏
-
369 收藏
-
280 收藏
-
142 收藏
-
289 收藏
-
203 收藏
-
488 收藏
-
447 收藏
-
152 收藏
-
381 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习