当前位置:   article > 正文

python中asyncio异步编程教程2_loop.run_in_executor

loop.run_in_executor

1. asyncio中的Future对象

asyncio中的Future对象是一个相对更偏向底层的可等待对象,通常我们不会直接用到这个对象,而是直接使用Task对象来完成任务的并发和状态的追踪。( Task 是 Futrue的子类 )

Future为我们提供了异步编程中的 最终结果 的处理(Task类也具备状态处理的功能),Task对象内部await结果的处理基于Future对象来的。

示例1:

  1. import asyncio
  2. async def main():
  3. # 获取当前事件循环
  4. loop = asyncio.get_running_loop()
  5. # # 创建一个任务(Future对象),这个任务什么都不干。
  6. fut = loop.create_future()
  7. # 等待任务最终结果(Future对象),没有结果则会一直等下去。
  8. await fut
  9. asyncio.run(main())

示例2:

  1. import asyncio
  2. async def set_after(fut):
  3. await asyncio.sleep(2)
  4. fut.set_result("666")
  5. async def main():
  6. # 获取当前事件循环
  7. loop = asyncio.get_running_loop()
  8. # 创建一个任务(Future对象),没绑定任何行为,则这个任务永远不知道什么时候结束。
  9. fut = loop.create_future()
  10. # 创建一个任务(Task对象),绑定了set_after函数,函数内部在2s之后,会给fut赋值。
  11. # 即手动设置future任务的最终结果,那么fut就可以结束了。
  12. await loop.create_task(set_after(fut))
  13. # 等待 Future对象获取 最终结果,否则一直等下去
  14. data = await fut
  15. print(data)
  16. asyncio.run(main()

Future对象本身函数进行绑定,所以想要让事件循环获取Future的结果,则需要手动设置。而Task对象继承了Future对象,其实就对Future进行扩展,他可以实现在对应绑定的函数执行完成之后,自动执行set_result,从而实现自动结束。

虽然,平时使用的是Task对象,但对于结果的处理本质是基于Future对象来实现的。

扩展:支持 await 对象语 法的对象课成为可等待对象,所以 协程对象Task对象Future对象 都可以被成为可等待对象。

2. concurrent.futures.Future对象

在Python的concurrent.futures模块中也有一个Future对象,这个对象是基于线程池和进程池实现异步操作时使用的对象。

  1. import time
  2. from concurrent.futures import Future
  3. from concurrent.futures.thread import ThreadPoolExecutor
  4. from concurrent.futures.process import ProcessPoolExecutor
  5. def func(value):
  6. time.sleep(1)
  7. print(value)
  8. pool = ThreadPoolExecutor(max_workers=5)
  9. # 或 pool = ProcessPoolExecutor(max_workers=5)
  10. for i in range(10):
  11. fut = pool.submit(func, i)
  12. print(fut)

在Python提供了一个将futures.Future 对象包装成asyncio.Future对象的函数 asynic.wrap_future

接下里你肯定问:为什么python会提供这种功能?

其实,一般在程序开发中我们要么统一使用 asycio 的协程实现异步操作、要么都使用进程池和线程池实现异步操作。但如果 协程的异步和 进程池/线程池的异步 混搭时,那么就会用到此功能了。

  1. import time
  2. import asyncio
  3. import concurrent.futures
  4. def func1():
  5. # 某个耗时操作
  6. time.sleep(1)
  7. return "SB"
  8. async def main():
  9. # 获取当前事件循环列表
  10. loop = asyncio.get_running_loop()
  11. # 1. Run in the default loop's executor ( 默认ThreadPoolExecutor )
  12. # 第一步:内部会先调用 ThreadPoolExecutor 的 submit 方法去线程池中申请一个线程去执行func1函数,并返回一个concurrent.futures.Future对象
  13. # 第二步:调用asyncio.wrap_future将concurrent.futures.Future对象包装为asycio.Future对象。
  14. # 因为concurrent.futures.Future对象不支持await语法,所以需要包装为 asycio.Future对象 才能使用。
  15. fut = loop.run_in_executor(None, func1)
  16. result = await fut
  17. print('default thread pool', result)
  18. # 2. Run in a custom thread pool:
  19. # with concurrent.futures.ThreadPoolExecutor() as pool:
  20. # result = await loop.run_in_executor(
  21. # pool, func1)
  22. # print('custom thread pool', result)
  23. # 3. Run in a custom process pool:
  24. # with concurrent.futures.ProcessPoolExecutor() as pool:
  25. # result = await loop.run_in_executor(
  26. # pool, func1)
  27. # print('custom process pool', result)
  28. asyncio.run(main())

应用场景:当项目以协程式的异步编程开发时,如果要使用一个第三方模块,而第三方模块不支持协程方式异步编程时,就需要用到这个功能,例如:

  1. import asyncio
  2. import requests
  3. async def download_image(url):
  4. # 发送网络请求,下载图片(遇到网络下载图片的IO请求,自动化切换到其他任务)
  5. print("开始下载:", url)
  6. loop = asyncio.get_event_loop()
  7. # requests模块默认不支持异步操作,所以就使用线程池来配合实现了。
  8. future = loop.run_in_executor(None, requests.get, url)
  9. response = await future
  10. print('下载完成')
  11. # 图片保存到本地文件
  12. file_name = url.rsplit('_')[-1]
  13. with open("media/{}".format(file_name), mode='wb') as file_object:
  14. file_object.write(response.content)
  15. if __name__ == '__main__':
  16. url_list = [
  17. 'https://www3.autoimg.cn/newsdfs/g26/M02/35/A9/120x90_0_autohomecar__ChsEe12AXQ6AOOH_AAFocMs8nzU621.jpg',
  18. 'https://img-pre.ivsky.com/img/tupian/pre/202103/03/sunyunzhu_diaodaishan-001.jpg',
  19. 'https://img-pre.ivsky.com/img/tupian/pre/202103/01/sunyunzhu_changxiushan-012.jpg'
  20. ]
  21. tasks = [download_image(url) for url in url_list]
  22. # 创建时间循环列表
  23. loop = asyncio.get_event_loop()
  24. # 将task对象当做任务添加到事件循环列表中去
  25. loop.run_until_complete(asyncio.wait(tasks))

3. 异步迭代器

什么是异步迭代器

实现了 __aiter__() 和 __anext__() 方法的对象。__anext__ 必须返回一个 awaitable 对象。async for 会处理异步迭代器的 __anext__() 方法所返回的可等待对象,直到其引发一个 StopAsyncIteration 异常。

什么是异步可迭代对象?

可在 async for 语句中被使用的对象。必须通过它的 __aiter__() 方法返回一个 asynchronous iterator

  1. import asyncio
  2. class Reader(object):
  3. """ 自定义异步迭代器(同时也是异步可迭代对象) """
  4. def __init__(self):
  5. self.count = 0
  6. async def readline(self):
  7. # await asyncio.sleep(1)
  8. self.count += 1
  9. if self.count == 100:
  10. return None
  11. return self.count
  12. def __aiter__(self):
  13. return self
  14. async def __anext__(self):
  15. val = await self.readline()
  16. if val == None:
  17. raise StopAsyncIteration
  18. return val
  19. async def func():
  20. # 创建异步可迭代对象
  21. async_iter = Reader()
  22. # async for 必须要放在async def函数内,否则语法错误。
  23. async for item in async_iter:
  24. print(item)
  25. asyncio.run(func())

4. 异步上下文管理器

此种对象通过定义 __aenter__() 和 __aexit__() 方法来对 async with 语句中的环境进行控制

  1. import asyncio
  2. class AsyncContextManager:
  3. def __init__(self):
  4. self.conn = conn
  5. async def do_something(self):
  6. # 异步操作数据库
  7. return 666
  8. async def __aenter__(self):
  9. # 异步链接数据库
  10. self.conn = await asyncio.sleep(1)
  11. return self
  12. async def __aexit__(self, exc_type, exc, tb):
  13. # 异步关闭数据库链接
  14. await asyncio.sleep(1)
  15. async def func():
  16. async with AsyncContextManager() as f:
  17. result = await f.do_something()
  18. print(result)
  19. asyncio.run(func())

这个异步的上下文管理器还是比较有用的,平时在开发过程中 打开、处理、关闭 操作时,就可以用这种方式来处理。

5. uvloop

uvloop是 asyncio 中的事件循环的替代方案,替换后可以使得asyncio性能提高。事实上,uvloop要比nodejs、gevent等其他python异步框架至少要快2倍,性能可以比肩Go语言。

安装uvloop: pip install uvloop

在项目中想要使用uvloop替换asyncio的事件循环也非常简单,只要在代码中这么做就行。

  1. import asyncio
  2. import uvloop
  3. asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
  4. # 编写asyncio的代码,与之前写的代码一致。
  5. # 内部的事件循环自动化会变为uvloop
  6. asyncio.run(...)

注意:知名的asgi uvicorn内部就是使用的uvloop的事件循环。

6. 总结:


在程序中只要看到async和await关键字,其内部就是基于协程实现的异步编程,这种异步编程是通过一个线程在IO等待时间去执行其他任务,从而实现并发。

如果是 I/O 密集型,且 I/O 请求比较耗时的话,使用协程。
如果是 I/O 密集型,且 I/O 请求比较快的话,使用多线程。
如果是 计算 密集型,考虑可以使用多核 CPU,使用多进程。

参考链接:https://www.cnblogs.com/wupeiqi/p/12834355.html
 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Gausst松鼠会/article/detail/102558
推荐阅读
相关标签
  

闽ICP备14008679号