赞
踩
asyncio中的Future对象是一个相对更偏向底层的可等待对象,通常我们不会直接用到这个对象,而是直接使用Task对象来完成任务的并发和状态的追踪。( Task 是 Futrue的子类 )
Future为我们提供了异步编程中的 最终结果 的处理(Task类也具备状态处理的功能),Task对象内部await结果的处理基于Future对象来的。
示例1:
- import asyncio
-
-
- async def main():
- # 获取当前事件循环
- loop = asyncio.get_running_loop()
- # # 创建一个任务(Future对象),这个任务什么都不干。
- fut = loop.create_future()
- # 等待任务最终结果(Future对象),没有结果则会一直等下去。
- await fut
- asyncio.run(main())
示例2:
- import asyncio
-
-
- async def set_after(fut):
- await asyncio.sleep(2)
- fut.set_result("666")
-
-
- async def main():
- # 获取当前事件循环
- loop = asyncio.get_running_loop()
- # 创建一个任务(Future对象),没绑定任何行为,则这个任务永远不知道什么时候结束。
- fut = loop.create_future()
- # 创建一个任务(Task对象),绑定了set_after函数,函数内部在2s之后,会给fut赋值。
- # 即手动设置future任务的最终结果,那么fut就可以结束了。
- await loop.create_task(set_after(fut))
- # 等待 Future对象获取 最终结果,否则一直等下去
- data = await fut
- print(data)
-
-
- asyncio.run(main()
Future对象本身函数进行绑定,所以想要让事件循环获取Future的结果,则需要手动设置。而Task对象继承了Future对象,其实就对Future进行扩展,他可以实现在对应绑定的函数执行完成之后,自动执行set_result
,从而实现自动结束。
虽然,平时使用的是Task对象,但对于结果的处理本质是基于Future对象来实现的。
扩展:支持 await 对象
语 法的对象课成为可等待对象,所以 协程对象
、Task对象
、Future对象
都可以被成为可等待对象。
在Python的concurrent.futures
模块中也有一个Future对象,这个对象是基于线程池和进程池实现异步操作时使用的对象。
- import time
- from concurrent.futures import Future
- from concurrent.futures.thread import ThreadPoolExecutor
- from concurrent.futures.process import ProcessPoolExecutor
- def func(value):
- time.sleep(1)
- print(value)
- pool = ThreadPoolExecutor(max_workers=5)
- # 或 pool = ProcessPoolExecutor(max_workers=5)
- for i in range(10):
- fut = pool.submit(func, i)
- print(fut)
在Python提供了一个将futures.Future
对象包装成asyncio.Future
对象的函数 asynic.wrap_future
。
接下里你肯定问:为什么python会提供这种功能?
其实,一般在程序开发中我们要么统一使用 asycio 的协程实现异步操作、要么都使用进程池和线程池实现异步操作。但如果 协程的异步
和 进程池/线程池的异步
混搭时,那么就会用到此功能了。
- import time
- import asyncio
- import concurrent.futures
-
- def func1():
- # 某个耗时操作
- time.sleep(1)
- return "SB"
-
- async def main():
- # 获取当前事件循环列表
- loop = asyncio.get_running_loop()
- # 1. Run in the default loop's executor ( 默认ThreadPoolExecutor )
- # 第一步:内部会先调用 ThreadPoolExecutor 的 submit 方法去线程池中申请一个线程去执行func1函数,并返回一个concurrent.futures.Future对象
- # 第二步:调用asyncio.wrap_future将concurrent.futures.Future对象包装为asycio.Future对象。
- # 因为concurrent.futures.Future对象不支持await语法,所以需要包装为 asycio.Future对象 才能使用。
- fut = loop.run_in_executor(None, func1)
- result = await fut
- print('default thread pool', result)
- # 2. Run in a custom thread pool:
- # with concurrent.futures.ThreadPoolExecutor() as pool:
- # result = await loop.run_in_executor(
- # pool, func1)
- # print('custom thread pool', result)
- # 3. Run in a custom process pool:
- # with concurrent.futures.ProcessPoolExecutor() as pool:
- # result = await loop.run_in_executor(
- # pool, func1)
- # print('custom process pool', result)
-
- asyncio.run(main())
应用场景:当项目以协程式的异步编程开发时,如果要使用一个第三方模块,而第三方模块不支持协程方式异步编程时,就需要用到这个功能,例如:
- import asyncio
- import requests
-
-
- async def download_image(url):
- # 发送网络请求,下载图片(遇到网络下载图片的IO请求,自动化切换到其他任务)
- print("开始下载:", url)
- loop = asyncio.get_event_loop()
- # requests模块默认不支持异步操作,所以就使用线程池来配合实现了。
- future = loop.run_in_executor(None, requests.get, url)
- response = await future
- print('下载完成')
- # 图片保存到本地文件
- file_name = url.rsplit('_')[-1]
- with open("media/{}".format(file_name), mode='wb') as file_object:
- file_object.write(response.content)
-
-
- if __name__ == '__main__':
- url_list = [
- 'https://www3.autoimg.cn/newsdfs/g26/M02/35/A9/120x90_0_autohomecar__ChsEe12AXQ6AOOH_AAFocMs8nzU621.jpg',
- 'https://img-pre.ivsky.com/img/tupian/pre/202103/03/sunyunzhu_diaodaishan-001.jpg',
- 'https://img-pre.ivsky.com/img/tupian/pre/202103/01/sunyunzhu_changxiushan-012.jpg'
- ]
- tasks = [download_image(url) for url in url_list]
- # 创建时间循环列表
- loop = asyncio.get_event_loop()
- # 将task对象当做任务添加到事件循环列表中去
- loop.run_until_complete(asyncio.wait(tasks))
什么是异步迭代器
实现了 __aiter__()
和 __anext__()
方法的对象。__anext__
必须返回一个 awaitable 对象。async for
会处理异步迭代器的 __anext__()
方法所返回的可等待对象,直到其引发一个 StopAsyncIteration
异常。
什么是异步可迭代对象?
可在 async for
语句中被使用的对象。必须通过它的 __aiter__()
方法返回一个 asynchronous iterator。
- import asyncio
-
-
- class Reader(object):
- """ 自定义异步迭代器(同时也是异步可迭代对象) """
-
- def __init__(self):
- self.count = 0
-
- async def readline(self):
- # await asyncio.sleep(1)
- self.count += 1
- if self.count == 100:
- return None
- return self.count
-
- def __aiter__(self):
- return self
-
- async def __anext__(self):
- val = await self.readline()
- if val == None:
- raise StopAsyncIteration
- return val
-
-
- async def func():
- # 创建异步可迭代对象
- async_iter = Reader()
- # async for 必须要放在async def函数内,否则语法错误。
- async for item in async_iter:
- print(item)
-
-
- asyncio.run(func())
此种对象通过定义 __aenter__()
和 __aexit__()
方法来对 async with
语句中的环境进行控制
- import asyncio
- class AsyncContextManager:
- def __init__(self):
- self.conn = conn
- async def do_something(self):
- # 异步操作数据库
- return 666
- async def __aenter__(self):
- # 异步链接数据库
- self.conn = await asyncio.sleep(1)
- return self
- async def __aexit__(self, exc_type, exc, tb):
- # 异步关闭数据库链接
- await asyncio.sleep(1)
- async def func():
- async with AsyncContextManager() as f:
- result = await f.do_something()
- print(result)
- asyncio.run(func())
这个异步的上下文管理器还是比较有用的,平时在开发过程中 打开、处理、关闭 操作时,就可以用这种方式来处理。
uvloop是 asyncio 中的事件循环的替代方案,替换后可以使得asyncio性能提高。事实上,uvloop要比nodejs、gevent等其他python异步框架至少要快2倍,性能可以比肩Go语言。
安装uvloop: pip install uvloop
在项目中想要使用uvloop替换asyncio的事件循环也非常简单,只要在代码中这么做就行。
- import asyncio
- import uvloop
- asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
- # 编写asyncio的代码,与之前写的代码一致。
- # 内部的事件循环自动化会变为uvloop
- asyncio.run(...)
注意:知名的asgi uvicorn内部就是使用的uvloop的事件循环。
在程序中只要看到async和await关键字,其内部就是基于协程实现的异步编程,这种异步编程是通过一个线程在IO等待时间去执行其他任务,从而实现并发。
如果是 I/O 密集型,且 I/O 请求比较耗时的话,使用协程。
如果是 I/O 密集型,且 I/O 请求比较快的话,使用多线程。
如果是 计算 密集型,考虑可以使用多核 CPU,使用多进程。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。