概念:协程(Coroutine),是一种线程内部的任务调度机制,它通过事件循环,在用户态中实现任务的挂起与恢复执行,从而在遇到 IO 操作时,不让 CPU 等待,而是继续执行其他需要 CPU 的任务。
协程的本质就是:在一个线程里,趁着某些任务在等 IO,把 CPU 交给其他任务去用。
关键点:
➊ 协程不是线程,也不是进程
➋ 协程发生在一个线程内部
➌ 协程的核心能力:挂起与恢复
➍ 协程依赖一个关键角色:事件循环
➎ 协程的目的是尽量减少线程切换
协程函数(coroutine Function):使用『async关键字』修饰的函数,就是协程函数。
协程对象(coroutine Object):调用『协程函数』,就会得到『协程对象』。
注意:调用『协程函数』,并不会执行『协程函数』中的代码。
# 定义一个协程函数
async def work():
print('work开始')
print('work执行中……')
print('work结束')
return '工作结果'
# 调用协程函数,会得到协程对象
coroutine_obj = work()
print(coroutine_obj) # <coroutine object work at 0x0000021AFC745B40>
将协程对象交给asyncio.run(),asyncio.run()会将协程对象包装成一个任务交给事件循环。
import asyncio
# 定义一个协程函数
async def work():
print('work开始')
print('work执行中……')
print('work结束')
return '工作结果'
# 调用协程函数,会得到协程对象
coroutine_obj = work()
print(coroutine_obj) # <coroutine object work at 0x0000021AFC745B40>
result = asyncio.run(coroutine_obj)
print(result)
代码解析asyncio.run() 方法做了3件事:
注意:asyncio.run() 方法会阻塞当前线程,直到任务执行完毕,并返回该任务 return 的最终结果。
await 关键字有以下三个作用:
➊ 挂起:await 会暂停当前协程的执行。
➋ 等待:遇到 await 关键字,事件循环会立即安排 await 后面的对象去执行,并等待该对象执行完成,并且可以拿到执行结果。
➌ 恢复:当 await 后的对象执行完毕,事件循环会恢复之前被挂起的协程,该协程会从当时挂起的位置继续执行,并拿到返回值。
【关键点】在执行 await 后面的对象时,会出现两种情况:
⚠️注意:await 后面只能写『可等待对象』,常见可等待对象有:协程对象、Future 对象、Task 对象。
import asyncio
async def work():
print('work开始')
print('work执行中……')
# await去等待一个协程对象(靠asyncio.sleep方法,返回一个协程对象)
await asyncio.sleep(2)
print('work结束')
return '工作结果'
async def main():
print('main开始')
# await去等待一个协程对象(靠自己去编写协程函数,随后调用该函数来得到协程对象)
res = await work()
print(res)
print('main结束')
return 'main的返回值'
result = asyncio.run(main())
print(result)
使用 await 实现多个任务同步执行。
import asyncio, time
async def work(n, delay):
print(f'{'\n' if n > 1 else ''}work{n}开始')
print(f'work{n}执行中……')
# 模拟一个IO等待
await asyncio.sleep(delay)
print(f'work{n}结束')
return f'work{n}的返回值'
async def main():
print('=== main开始')
start = time.time()
# 调用三次work函数,分别得到三个协程对象
coroutine1 = work(1, 2)
coroutine2 = work(2, 2)
coroutine3 = work(3, 2)
# 此处会等待coroutine1执行完成
res1 = await coroutine1
print(res1)
# 等待上面的coroutine1完成后,再等待coroutine2完成
res2 = await coroutine2
print(res2)
# 等待上面的coroutine2完成后,再等待coroutine3完成
res3 = await coroutine3
print(res3)
print('main结束 ====', f'耗时:{time.time() - start}秒') # 耗时:6.012717247009277秒
return '*** main的返回值'
# 将协程对象交给事件循环
result = asyncio.run(main())
print(result)
使用asyncio.create_task()方法向事件循环中添加任务,从而实现多个任务异步执行。
import asyncio, time
async def work(n, delay):
print(f'{'\n' if n > 1 else ''}work{n}开始')
print(f'work{n}执行中……')
# 模拟一个IO等待
await asyncio.sleep(delay)
print(f'work{n}结束')
return f'work{n}的返回值'
async def main():
print('=== main开始')
start = time.time()
# asyncio.create_task 会把一个协程对象包装成一个可被事件循环调度的任务,并注册到事件循环中
task1 = asyncio.create_task(work(1, 2))
task2 = asyncio.create_task(work(2, 2))
task3 = asyncio.create_task(work(3, 2))
# 此处会等待task1执行完成
res1 = await task1
print(res1)
# 等待上面的task1完成后,再等待task2完成
res2 = await task2
print(res2)
# 等待上面的task2完成后,再等待task3完成
res3 = await task3
print(res3)
print('main结束 ====', f'耗时:{time.time() - start}秒') # 耗时:6.012717247009277秒
return '*** main的返回值'
# 将协程对象交给事件循环
result = asyncio.run(main())
print(result)
asyncio.gather()方法可以把多个协程对象丢给事件循环,并在全部执行完成后,一次性拿到所有结果。
import asyncio, time
async def work(n, delay):
print(f'{'\n' if n > 1 else ''}work{n}开始')
print(f'work{n}执行中……')
# 模拟一个IO等待
await asyncio.sleep(delay)
print(f'work{n}结束')
return f'work{n}的返回值'
async def main():
print('=== main开始')
start = time.time()
# 把多个协程对象同时丢给事件循环,并在全部执行完后,一次性拿到所有结果。
result = await asyncio.gather(work(1, 2), work(2, 2), work(3, 2))
print(result)
print('main结束 ====', f'耗时:{time.time() - start}秒') # 耗时:6.012717247009277秒
return '*** main的返回值'
# 将协程对象交给事件循环
result = asyncio.run(main())
print(result)
➊ 使用传统方式下载图片
传统方式的特点是:图片是一张一张下载的,当前图片没有下载完成,后一张图片的下载就不能开始,这属于典型的同步下载。
首先需要安装软件包requests
pip install requests
import requests
def download_picture(url):
print(f'开始下载: {url}')
# 下载
response = requests.get(url)
print('下载完毕')
# 保存到本地
with open(f'./media/{url.split('/')[-1]}', 'wb') as f:
f.write(response.content)
def main():
url_list = [
'https://n.sinaimg.cn/spider20260129/217/w600h417/20260129/3e26-917ee55a8a42b8626807c332c24981de.png',
'https://n.sinaimg.cn/finance/transform/97/w630h267/20260129/97c4-b211cc51784830f09ee19e450475c93b.png',
'https://n.sinaimg.cn/spider20260129/539/w1439h700/20260129/e09a-cc2ca319e00f701ccfca3ebc62aa8772.png'
]
for url in url_list:
download_picture(url)
main()
➋ 使用协程方式下载图片
协程方式的特点是:多张图片会几乎同时发起下载请求,当某一张图片在等待网络数据返回时,其它图片的下载任务并不会被阻塞,而是可以继续执行,这属于典型的协程并发下载。
首先需要安装软件包aiohttp
pip install aiohttp
import aiohttp
import asyncio
async def download_picture(session, url):
print(f'开始下载:{url}')
# 发送网络请求,获取这张图片,请求发出去后,要等待服务器把数据返回,等的这段时间就是IO等待
response = await session.get(url)
# 等待数据(图片数据可能分多次传输,需要等待数据全部读完,等的这段时间也是IO等待)
content = await response.read()
print('下载完毕')
# 保存图片到本地
with open(f'./media/{url[-10:]}', 'wb') as file:
file.write(content)
# 释放连接资源(告诉 aiohttp,这个连接我不用了,你可以回收了)
await response.release()
async def main():
url_list = [
'https://n.sinaimg.cn/spider20260129/217/w600h417/20260129/3e26-917ee55a8a42b8626807c332c24981de.png',
'https://n.sinaimg.cn/finance/transform/97/w630h267/20260129/97c4-b211cc51784830f09ee19e450475c93b.png',
'https://n.sinaimg.cn/spider20260129/539/w1439h700/20260129/e09a-cc2ca319e00f701ccfca3ebc62aa8772.png'
]
# 创建会话对象(发请求的工具)
session = aiohttp.ClientSession()
# 创建多个协程对象
coroutine_list = [download_picture(session, url) for url in url_list]
# 将多个协程对象交给事件循环
await asyncio.gather(*coroutine_list)
# 关闭会话
await session.close()
asyncio.run(main())
🔥BuildAdmin是一个永久免费开源,无需授权即可商业使用,且使用了流行技术栈快速创建商业级后台管理系统。