python异步asy_Python 异步编程之asyncio【转载】

一、协程的认识

协程(Coroutine),也可以被称为微线程,是一种用户态内的上下文切换技术。

简而言之,其实就是通过一个线程实现代码块相互切换执行。例如:deffunc1():print(1)

...print(2)deffunc2():print(3)

...print(4)

func1()

func2()

上面代码可以看出来,是串行的,如果有一个函数阻塞,则其它函要等待,导致时间浪费,这个就是协程存在的原因。

二、操作协程的模块有哪些呢?

在Python中有多种方式可以实现协程,例如:

1、greenlet,是一个第三方模块,用于实现协程代码(Gevent协程就是基于greenlet实现)2、yield,生成器,借助生成器的特点也可以实现协程代码。

3、asyncio,在Python3.4中引入的模块用于编写协程代码。

4、async& awiat,在Python3.5中引入的两个关键字,结合asyncio模块可以更方便的编写协程代码。

1、greenlet示例

ContractedBlock.gif

ExpandedBlockStart.gif

from greenlet importgreenletdeffunc1():print(1) #第1步:输出 1

gr2.switch() #第3步:切换到 func2 函数

print(2) #第6步:输出 2

gr2.switch() #第7步:切换到 func2 函数,从上一次执行的位置继续向后执行

deffunc2():print(3) #第4步:输出 3

gr1.switch() #第5步:切换到 func1 函数,从上一次执行的位置继续向后执行

print(4) #第8步:输出 4

if __name__ == '__main__':

gr1=greenlet(func1)

gr2=greenlet(func2)

gr1.switch()#第1步:去执行 func1 函数

#输出结果

#1

#3

#2

#4

注意:switch中也可以传递参数用于在切换执行时相互传递值。

greenlet示例

2、yield示例

ContractedBlock.gif

ExpandedBlockStart.gif

deffunc1():yield 1

yield fromfunc2()yield 2

deffunc2():yield 3

yield 4

if __name__ == '__main__':

f1=func1()for item inf1:print(item)#输出结果

#1

#3

#4

#2

注意:yield form关键字是在Python3.3中引入的

yield示例

3、asyncio示例

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

@asyncio.coroutinedeffunc1():print(1)yield from asyncio.sleep(2) #遇到IO耗时操作,自动化切换到tasks中的其他任务

print(2)

@asyncio.coroutinedeffunc2():print(3)yield from asyncio.sleep(2) #遇到IO耗时操作,自动化切换到tasks中的其他任务

print(4)if __name__ == '__main__':

tasks=[

asyncio.ensure_future(func1()),

asyncio.ensure_future(func2())

]

loop=asyncio.get_event_loop()

loop.run_until_complete(asyncio.wait(tasks))

注意:

基于asyncio模块实现的协程比之前的要更厉害,因为他的内部还集成了遇到IO耗时操作自动切花的功能。

在Python3.4之前官方未提供协程的类库,一般大家都是使用greenlet等其他来实现。在Python3.4发布后官方正式支持协程,即:asyncio模块。

asyncio示例

4、async & await示例

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc1():print(1)

await asyncio.sleep(2)print(2)

asyncdeffunc2():print(3)

await asyncio.sleep(2)print(4)if __name__ == '__main__':

tasks=[

asyncio.ensure_future(func1()),

asyncio.ensure_future(func2())

]

loop=asyncio.get_event_loop()

loop.run_until_complete(asyncio.wait(tasks))#输出结果

#1

#3

#2

#4

注意:

async&awit 关键字在Python3.5版本中正式引入,基于他编写的协程代码其实就是 上一示例 的加强版,让代码可以更加简便。

Python3.8之后 @asyncio.coroutine 装饰器就会被移除,推荐使用async& awit 关键字实现协程代码。

async & await示例

5、小结

关于协程有多种实现方式,目前主流使用是Python官方推荐的asyncio模块和async&await关键字的方式,

例如:在tonado、sanic、fastapi、django3 中均已支持。

三、协程的意义

通过学习,我们已经了解到协程可以通过一个线程在多个上下文中进行来回切换执行。

但是,协程来回切换执行的意义何在呢?

(1)、计算型的操作,利用协程来回切换执行,没有任何意义,来回切换并保存状态 反倒会降低性能。

(2)、IO型的操作,利用协程在IO等待时间就去切换执行其他任务,当IO操作结束后再自动回调,那么就会大大节省资源并提供性能,

从而实现异步编程(不等待任务结束就可以去执行其他代码)。

1、爬虫案例

1.1、同步爬虫案例

ContractedBlock.gif

ExpandedBlockStart.gif

importrequestsdefdownload_image(url):print("开始下载:", url)#发送网络请求,下载图片

response =requests.get(url)print("下载完成")#图片保存到本地文件

file_name = url.rsplit('_')[-1]

with open(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://www2.autoimg.cn/newsdfs/g30/M01/3C/E2/120x90_0_autohomecar__ChcCSV2BBICAUntfAADjJFd6800429.jpg','https://www3.autoimg.cn/newsdfs/g26/M0B/3C/65/120x90_0_autohomecar__ChcCP12BFCmAIO83AAGq7vK0sGY193.jpg']for item inurl_list:

download_image(item)

注意:请提前安装:pip3 install requests

同步爬虫案例

1.2、异步爬虫案例

ContractedBlock.gif

ExpandedBlockStart.gif

importaiohttpimportasyncio

asyncdeffetch(session, url):print("发送请求:", url)

async with session.get(url, verify_ssl=False) as response:

content=await response.content.read()

file_name= url.rsplit('_')[-1]

with open(file_name, mode='wb') as file_object:

file_object.write(content)

asyncdefmain():

async with aiohttp.ClientSession() as session:

url_list=['https://www3.autoimg.cn/newsdfs/g26/M02/35/A9/120x90_0_autohomecar__ChsEe12AXQ6AOOH_AAFocMs8nzU621.jpg','https://www2.autoimg.cn/newsdfs/g30/M01/3C/E2/120x90_0_autohomecar__ChcCSV2BBICAUntfAADjJFd6800429.jpg','https://www3.autoimg.cn/newsdfs/g26/M0B/3C/65/120x90_0_autohomecar__ChcCP12BFCmAIO83AAGq7vK0sGY193.jpg']

tasks= [asyncio.create_task(fetch(session, url)) for url inurl_list]

await asyncio.wait(tasks)if __name__ == '__main__':

asyncio.run(main())

注意:请提前安装:pip3 install aiohttp

异步爬虫案例

1.3、同步编程和异常编程的对比

上述两种的执行对比之后会发现,基于协程的异步编程 要比 同步编程的效率高了很多。因为:

同步编程,按照顺序逐一排队执行,如果图片下载时间为2分钟,那么全部执行完则需要6分钟。

异步编程,几乎同时发出了3个下载任务的请求(遇到IO请求自动切换去发送其他任务请求),如果图片下载时间为2分钟,那么全部执行完毕也大概需要2分钟左右就可以了。

1.4、总结

协程一般应用在有IO操作的程序中,因为协程可以利用IO等待的时间去执行一些其他的代码,从而提升代码执行效率。

生活中不也是这样的么,假设 你是一家制造汽车的老板,员工点击设备的【开始】按钮之后,在设备前需等待30分钟,

然后点击【结束】按钮,此时作为老板的你一定希望这个员工在等待的那30分钟的时间去做点其他的工作。

四、异步编程

基于async &await关键字的协程可以实现异步编程,这也是目前python异步相关的主流技术。

想要真正的了解Python中内置的异步编程,根据下文的顺序一点点来看。

1、事件循环的概述

事件循环,可以把他当做是一个while循环,这个while循环在周期性的运行并执行一些任务,在特定条件下终止循环。在编写程序时候可以通过如下代码来获取和创建事件循环。importasyncio

loop= asyncio.get_event_loop()

ContractedBlock.gif

ExpandedBlockStart.gif

#伪代码

任务列表 =[ 任务1, 任务2, 任务3,... ]whileTrue:

可执行的任务列表,已完成的任务列表= 去任务列表中检查所有的任务,将'可执行'和'已完成'的任务返回for 就绪任务 in已准备就绪的任务列表:

执行已就绪的任务for 已完成的任务 in已完成的任务列表:

在任务列表中移除 已完成的任务

如果 任务列表 中的任务都已完成,则终止循环

伪代码

2、基本应用的示例

程序中,如果想要执行协程函数的内部代码,需要 事件循环 和 协程对象 配合才能实现,如:

2.1、方式一、手动创建事件循环

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc():print("协程内部代码")if __name__ == '__main__':#调用协程函数,返回一个协程对象。

result =func()

loop= asyncio.get_event_loop() #创建一个事件循环

loop.run_until_complete(result) #将协程当做任务提交到事件循环的任务列表中,协程执行完成之后终止。

get_event_loop获取事件循环

2.2、方式二、async.run创建 事件循环

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc():print("协程内部代码")if __name__ == '__main__':#调用协程函数,返回一个协程对象。

result =func()#本质上方式一是一样的,内部先 创建事件循环 然后执行 run_until_complete,一个简便的写法。

#asyncio.run 函数在 Python 3.7 中加入 asyncio 模块,

asyncio.run(result)

async.run创建 事件循环

3、await

await是一个只能在协程函数中使用的关键字,用于遇到IO操作时挂起 当前协程(任务),当前协程(任务)挂起过程中 事件循环可以去执行其他的协程(任务),

当前协程IO处理完成时,可以再次切换回来执行await之后的代码。代码如下:

3.1、示例1、单任务等待IO再运行

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc():print("执行协程函数内部代码")#遇到IO操作挂起当前协程(任务),等IO操作完成之后再继续往下执行。

#当前协程挂起时,事件循环可以去执行其他协程(任务)。

response = await asyncio.sleep(2)print("IO请求结束,结果为:", response)

result=func()

asyncio.run(result)

示例1

3.2、示例2、遇到IO阻塞切至其它任务运行

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdefothers():print("start")

await asyncio.sleep(2)print('end')return '返回值'asyncdeffunc():print("执行协程函数内部代码")#遇到IO操作挂起当前协程(任务),等IO操作完成之后再继续往下执行。当前协程挂起时,事件循环可以去执行其他协程(任务)。

response =await others()print("IO请求结束,结果为:", response)

asyncio.run(func())

示例2

3.3、示例3、同一个协程函数可以调用多个协程函数

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdefothers():print("start")

await asyncio.sleep(2)print('end')return '返回值'asyncdeffunc():print("执行协程函数内部代码")#遇到IO操作挂起当前协程(任务),等IO操作完成之后再继续往下执行。当前协程挂起时,事件循环可以去执行其他协程(任务)。

response1 =await others()print("IO请求结束,结果为:", response1)

response2=await others()print("IO请求结束,结果为:", response2)

asyncio.run(func())

示例3

3.4、总结

上述的所有示例都只是创建了一个任务,即:事件循环的任务列表中只有一个任务,所以在IO等待时无法演示切换到其他任务效果。

在程序想要创建多个任务对象,需要使用Task对象来实现。

4、Task对象

官方文档:

Tasks are used to schedule coroutines concurrently.

When a coroutineis wrapped into a Task with functions like asyncio.create_task() the coroutine isautomatically scheduled to run soon。

翻译:

Tasks用于并发调度协程,通过asyncio.create_task(协程对象)的方式创建Task对象,这样可以让协程加入事件循环中等待被调度执行。

除了使用 asyncio.create_task() 函数以外,还可以用低层级的 loop.create_task() 或 ensure_future() 函数。不建议手动实例化 Task 对象。

本质上是将协程对象封装成task对象,并将协程立即加入事件循环,同时追踪协程的状态。

注意:asyncio.create_task() 函数在 Python3.7 中被加入。在 Python 3.7 之前,可以改用低层级的 asyncio.ensure_future() 函数。

4.1、示例1、单个task单个task创建的示例

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc():print(1)

await asyncio.sleep(2)print(2)return "返回值"asyncdefmain():print("main开始")#创建协程,将协程封装到一个Task对象中并立即添加到事件循环的任务列表中,等待事件循环去执行(默认是就绪状态)。

task1 =asyncio.create_task(func())#创建协程,将协程封装到一个Task对象中并立即添加到事件循环的任务列表中,等待事件循环去执行(默认是就绪状态)。

task2 =asyncio.create_task(func())print("main结束")#当执行某协程遇到IO操作时,会自动化切换执行其他任务。

#此处的await是等待相对应的协程全都执行完毕并获取结果

ret1 =await task1

ret2=await task2print(ret1, ret2)

asyncio.run(main())

示例1

4.2、示例2、将多个create_task任务放在列表中执行

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc():print(1)

await asyncio.sleep(2)print(2)return "返回值"asyncdefmain():print("main开始")#创建协程,将协程封装到Task对象中并添加到事件循环的任务列表中,等待事件循环去执行(默认是就绪状态)。

#在调用

task_list =[

asyncio.create_task(func()),

asyncio.create_task(func())

]print("main结束")#当执行某协程遇到IO操作时,会自动化切换执行其他任务。

#此处的await是等待所有协程执行完毕,并将所有协程的返回值保存到done

#如果设置了timeout值,则意味着此处最多等待的秒,完成的协程返回值写入到done中,未完成则写到pending中。

done, pending = await asyncio.wait(task_list, timeout=None)print(done, pending)

asyncio.run(main())

示例2

注意:asyncio.wait 源码内部会对列表中的每个协程执行ensure_future从而封装为Task对象,所以在和wait配合使用时task_list的值为[func(),func()] 也是可以的

4.3、示例3、将多个任务直接放在列表中执行

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdeffunc():print("执行协程函数内部代码")#遇到IO操作挂起当前协程(任务),等IO操作完成之后再继续往下执行。当前协程挂起时,事件循环可以去执行其他协程(任务)。

response = await asyncio.sleep(2)print("IO请求结束,结果为:", response)

coroutine_list=[func(), func()]#错误:coroutine_list = [ asyncio.create_task(func()), asyncio.create_task(func()) ]#此处不能直接 asyncio.create_task,因为将Task立即加入到事件循环的任务列表,#但此时事件循环还未创建,所以会报错。#使用asyncio.wait将列表封装为一个协程,并调用asyncio.run实现执行两个协程#asyncio.wait内部会对列表中的每个协程执行ensure_future,封装为Task对象。

done, pending = asyncio.run(asyncio.wait(coroutine_list))

示例3

5、asyncio.Future对象

官文描述:

A Futureis a special low-level awaitable object that represents an eventual result of an asynchronous operation.

翻译:

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

Future为我们提供了异步编程中的 最终结果 的处理(Task类也具备状态处理的功能)。

5.1、示例1、不跑任何任务运行Future

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdefmain():#获取当前事件循环

loop =asyncio.get_running_loop()## 创建一个任务(Future对象),这个任务什么都不干。

fut =loop.create_future()#等待任务最终结果(Future对象),没有结果则会一直等下去。

await fut

asyncio.run(main())

示例1

5.2、示例2、 给Future设置任务运行

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncio

asyncdefset_after(fut):

await asyncio.sleep(2)

fut.set_result("666")

asyncdefmain():#获取当前事件循环

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 futprint(data)

asyncio.run(main())

示例2

5.3、总结

Future对象本身函数进行绑定,所以想要让事件循环获取Future的结果,则需要手动设置。而Task对象继承了Future对象,

其实就对Future进行扩展,他可以实现在对应绑定的函数执行完成之后,自动执行set_result,从而实现自动结束。

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

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

6、futures.Future对象

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

ContractedBlock.gif

ExpandedBlockStart.gif

importtimefrom concurrent.futures importFuturefrom concurrent.futures.thread importThreadPoolExecutorfrom concurrent.futures.process importProcessPoolExecutordeffunc(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)"""运行结果:

0

2

1

4

3

7

6

9

8

5"""

Future对象的示例

两个Future对象是不同的,他们是为不同的应用场景而设计,例如:concurrent.futures.Future不支持await语法 等。

官方提示两对象之间不同:

unlike asyncio Futures, concurrent.futures.Future instances cannot be awaited.

asyncio.Future.result()and asyncio.Future.exception() do notaccept the timeout argument.

asyncio.Future.result()and asyncio.Future.exception() raise an InvalidStateError exception when the Future is notdone.

Callbacks registered with asyncio.Future.add_done_callback() arenotcalled immediately. They are scheduled with loop.call_soon() instead.

asyncio Futureis not compatible with the concurrent.futures.wait() andconcurrent.futures.as_completed() functions.

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

一般在程序开发中我们要么统一使用 asycio 的协程实现异步操作、要么都使用进程池和线程池实现异步操作。

但如果协程的异步和进程池/线程池的异步混搭时,那么就会用到此功能了。

6.1、利用asyncio/futures.ThreadPoolExecutor/futures.ProcessPoolExecutor 将非协程函数转为Future对象,再给事件循环异步

ContractedBlock.gif

ExpandedBlockStart.gif

importtimeimportasyncioimportconcurrent.futuresdeffunc1():#某个耗时操作

time.sleep(2)return "SB"asyncdefmain():

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 futprint('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())

普通函数转Future对象的三种方法示例

6.2、应用的示例,当遇到模块不支持协程时的转换示例

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncioimportrequests

asyncdefdownload_image(url):#发送网络请求,下载图片(遇到网络下载图片的IO请求,自动化切换到其他任务)

print("开始下载:", url)

loop=asyncio.get_event_loop()#requests模块默认不支持异步操作,所以就使用线程池来配合实现了。

future =loop.run_in_executor(None, requests.get, url)

response=await futureprint('下载完成')#图片保存到本地文件

file_name = url.rsplit('_')[-1]

with open(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://www2.autoimg.cn/newsdfs/g30/M01/3C/E2/120x90_0_autohomecar__ChcCSV2BBICAUntfAADjJFd6800429.jpg','https://www3.autoimg.cn/newsdfs/g26/M0B/3C/65/120x90_0_autohomecar__ChcCP12BFCmAIO83AAGq7vK0sGY193.jpg']

tasks= [download_image(url) for url inurl_list]

loop=asyncio.get_event_loop()

loop.run_until_complete(asyncio.wait(tasks))

requests模块转为异步的运行示例

7、异步迭代器

7.1、什么是异步迭代器?

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

直到其引发一个 StopAsyncIteration 异常。由 PEP 492 引入。

7.2、什么是异步可迭代对象?

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

asynchronous iterator。由 PEP 492 引入。

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncioclassReader(object):"""自定义异步迭代器(同时也是异步可迭代对象)"""

def __init__(self):

self.count=0

asyncdefreadline(self):#await asyncio.sleep(1)

self.count += 1

if self.count == 100:returnNonereturnself.countdef __aiter__(self):returnself

asyncdef __anext__(self):

val=await self.readline()if val ==None:raiseStopAsyncIterationreturnval

asyncdeffunc():#创建异步可迭代对象

async_iter =Reader()#async for 必须要放在async def函数内,否则语法错误。

async for item inasync_iter:print(item)

asyncio.run(func())

注意:异步迭代器其实没什么太大的作用,只是支持了async for语法而已。

异步迭代器的示例

7.3、异步上下文管理器

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

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncioclassAsyncContextManager:def __init__(self, conn):

self.conn=conn

asyncdefdo_something(self):#异步操作数据库

return 666asyncdef __aenter__(self):#异步链接数据库

self.conn = await asyncio.sleep(1)returnself

asyncdef __aexit__(self, exc_type, exc, tb):#异步关闭数据库链接

await asyncio.sleep(1)

asyncdeffunc():

conn=None

async with AsyncContextManager(conn) as f:

result=await f.do_something()print(result)

asyncio.run(func())

异步上下文管理器的示例

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

7.4、总结

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

8、uvloop

Python标准库中提供了asyncio模块,用于支持基于协程的异步编程。

uvloop是 asyncio 中的事件循环的替代方案,替换后可以使得asyncio性能提高。

事实上,uvloop要比nodejs、gevent等其他python异步框架至少要快2倍,性能可以比肩Go语言。

8.1、安装uvloop

pip3 install uvloop

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

ContractedBlock.gif

ExpandedBlockStart.gif

importasyncioimportuvloop

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

asyncio.run(...)

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

uvloop的使用示例

五、实战

为了更好理解,下面所有示例的IO情况都是以 asyncio.sleep 为例,而真实的项目开发中会用到很多IO的情况。

1、异步Redis

当通过python去操作redis时,链接、设置值、获取值 这些都涉及网络IO请求,使用asycio异步的方式可以在IO等待时去做一些其他任务,从而提升性能。

安装Python异步操作redis模块

pip3 install aioredis

1.1、示例1:异步操作redis

ContractedBlock.gif

ExpandedBlockStart.gif

#!/usr/bin/env python#-*- coding:utf-8 -*-

importasyncioimportaioredis

asyncdefexecute(address, password):print("开始执行", address)#网络IO操作:创建redis连接

redis = await aioredis.create_redis(address, password=password, db=3)#网络IO操作:在redis中设置哈希值car,内部在设三个键值对,即: redis = { car:{key1:1,key2:2,key3:3}}

await redis.hmset_dict('car', key1=1, key2=2, key3=3)#网络IO操作:去redis中获取值

result = await redis.hgetall('car', encoding='utf-8')print(result)

redis.close()#网络IO操作:关闭redis连接

await redis.wait_closed()print("结束", address)

asyncio.run(execute('redis://192.168.2.129:6379', "redis"))

异步操作redis的示例

1.2、示例2:连接多个redis做操作(遇到IO会切换其他任务,提供了性能)

ContractedBlock.gif

ExpandedBlockStart.gif

#!/usr/bin/env python#-*- coding:utf-8 -*-

importasyncioimportaioredis

asyncdefexecute(address, password):print("开始执行", address)#网络IO操作:先去连接 47.93.4.197:6379,遇到IO则自动切换任务,去连接47.93.4.198:6379

redis = await aioredis.create_redis_pool(address, password=password)#网络IO操作:遇到IO会自动切换任务

await redis.hmset_dict('car', key1=1, key2=2, key3=3)#网络IO操作:遇到IO会自动切换任务

result = await redis.hgetall('car', encoding='utf-8')print(result)

redis.close()#网络IO操作:遇到IO会自动切换任务

await redis.wait_closed()print("结束", address)

task_list=[

execute('redis://47.93.4.197:6379', "root!2345"),

execute('redis://47.93.4.198:6379', "root!2345")

]

asyncio.run(asyncio.wait(task_list))

异步多链接操作redis示例

2、异步MySQL

当通过python去操作MySQL时,连接、执行SQL、关闭都涉及网络IO请求,使用asycio异步的方式可以在IO等待时去做一些其他任务,从而提升性能。

安装Python异步操作redis模块

pip3 install aiomysql

2.1、示例1:连接数据库,发送一条查询语句查询数据

ContractedBlock.gif

ExpandedBlockStart.gif

#!/usr/bin/env python#-*- coding:utf-8 -*-

importasyncioimportaiomysql

asyncdefexecute():#网络IO操作:连接MySQL

conn = await aiomysql.connect(host='127.0.0.1', port=3306, user='root', password='123', db='mysql', )#网络IO操作:创建CURSOR

cur =await conn.cursor()#网络IO操作:执行SQL

await cur.execute("SELECT Host,User FROM user")#网络IO操作:获取SQL结果

result =await cur.fetchall()print(result)#网络IO操作:关闭链接

await cur.close()

conn.close()

asyncio.run(execute())

示例一

2.2、示例2:多连接数据库 ,查询数据库

ContractedBlock.gif

ExpandedBlockStart.gif

#!/usr/bin/env python#-*- coding:utf-8 -*-

importasyncioimportaiomysql

asyncdefexecute(host, password):print("开始", host)#网络IO操作:先去连接 47.93.40.197,遇到IO则自动切换任务,去连接47.93.40.198:6379

conn = await aiomysql.connect(host=host, port=3306, user='root', password=password, db='mysql')#网络IO操作:遇到IO会自动切换任务

cur =await conn.cursor()#网络IO操作:遇到IO会自动切换任务

await cur.execute("SELECT Host,User FROM user")#网络IO操作:遇到IO会自动切换任务

result =await cur.fetchall()print(result)#网络IO操作:遇到IO会自动切换任务

await cur.close()

conn.close()print("结束", host)

task_list=[

execute('47.93.40.197', "root!2345"),

execute('47.93.40.197', "root!2345")

]

asyncio.run(asyncio.wait(task_list))

示例二

3、FastAPI框架

FastAPI是一款用于构建API的高性能web框架,框架基于Python3.6+的 type hints搭建。

接下里的异步示例以FastAPI和uvicorn来讲解(uvicorn是一个支持异步的asgi)。

安装FastAPI web 框架

pip3 install fastapi

安装uvicorn,本质上为web提供socket server的支持的asgi(一般支持异步称asgi、不支持异步称wsgi)

pip3 install uvicorn

3.1、FastAPI + aioredis示例

ContractedBlock.gif

ExpandedBlockStart.gif

#!/usr/bin/env python#-*- coding:utf-8 -*-

importasyncioimportuvicornimportaioredisfrom aioredis importRedisfrom fastapi importFastAPI

app=FastAPI()

REDIS_POOL= aioredis.ConnectionsPool('redis://47.193.14.198:6379', password="root123", minsize=1, maxsize=10)

@app.get("/")defindex():"""普通操作接口"""

return {"message": "Hello World"}

@app.get("/red")

asyncdefred():"""异步操作接口"""

print("请求来了")

await asyncio.sleep(3)#连接池获取一个连接

conn =await REDIS_POOL.acquire()

redis=Redis(conn)#设置值

await redis.hmset_dict('car', key1=1, key2=2, key3=3)#读取值

result = await redis.hgetall('car', encoding='utf-8')print(result)#连接归还连接池

REDIS_POOL.release(conn)returnresultif __name__ == '__main__':

uvicorn.run("luffy:app", host="127.0.0.1", port=5000, log_level="info")

示例

在有多个用户并发请求的情况下,异步方式来编写的接口可以在IO等待过程中去处理其他的请求,提供性能。

例如:同时有两个用户并发来向接口 http://127.0.0.1:5000/red 发送请求,服务端只有一个线程,同一时刻只有一个请求被处理。

异步处理可以提供并发是因为:当视图函数在处理第一个请求时,第二个请求此时是等待被处理的状态,当第一个请求遇到IO等待时,会自动切换去接收并处理第二个请求,当遇到IO时自动化切换至其他请求,

一旦有请求IO执行完毕,则会再次回到指定请求向下继续执行其功能代码。

4、爬虫

在编写爬虫应用时,需要通过网络IO去请求目标数据,这种情况适合使用异步编程来提升性能,接下来我们使用支持异步编程的aiohttp模块来实现。

安装aiohttp模块

pip3 install aiohttp

ContractedBlock.gif

ExpandedBlockStart.gif

#!/usr/bin/env python#-*- coding:utf-8 -*-

importaiohttpimportasyncio

asyncdeffetch(session, url):print("发送请求:", url)

async with session.get(url, verify_ssl=False) as response:

text=await response.text()print("得到结果:", url, len(text))

asyncdefmain():

async with aiohttp.ClientSession() as session:

url_list=['https://python.org','https://www.baidu.com','https://www.pythonav.com']

tasks= [asyncio.create_task(fetch(session, url)) for url inurl_list]

await asyncio.wait(tasks)if __name__ == '__main__':

asyncio.run(main())

aiohttp爬虫示例

六、总结

为了提升性能越来越多的框架都在向异步编程靠拢,例如:sanic、tornado、django3.0、django channels组件 等,

用更少资源可以做处理更多的事,何乐而不为呢。

转载地址:https://www.cnblogs.com/wupeiqi/p/12834355.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/502935.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

bitcount java_Java源码解释之Integer.bitCount

Java中的Integer.bitCount(i)的返回值是i的二进制表示中1的个数。源码如下:public static int bitCount(int i) {// HD, Figure 5-2i i - ((i >>> 1) & 0x55555555);i (i & 0x33333333) ((i >>> 2) & 0x33333333);i (i (i >&…

git merge 冲突_卧槽!小姐姐用动画图解 Git 命令,这也太秀了吧?!

公众号关注 “GitHubDaily”设为 “星标”,每天带你逛 GitHub!大家好,我是小 G。在座的各位应该都知道,Git 作为居家必备、团队协作之利器,自从 Linus Torvalds 发布这款工具后,便一直受到各路开发者的喜爱…

android 删除文件 代码_代码审计之某系统后台存在任意删除文件

本文作者:霾团队交流群:673441920-----------------------------------------------------------前言POC镇楼!!!POST 漏洞演示过程:首先我们利用D盾监听下我们的项目以外的目录。这里刚刚我们创建了这个文件…

ubuntu java8 java9_在Ubuntu/Debian系统上安装Java 9的方法

本文介绍在Ubuntu/Debian系统上安装Oracle Java 9的方法:使用webupd8team/java PPA,相同的PPA提供了Java 8和Java 7等旧版Java的软件包,如果你的应用程序需要这个,可以随意安装它们。要安装新版本可参考在Ubuntu 18.04系统上安装J…

websocket 压力测试_打造最强移动测试平台

笔者今年换掉了服役N年的旧手机,新手机12G的RAM,比自用的本子内存都大,如果只是玩游戏感觉不能完全发挥出全部机能,但又因为怕影响日常使用没有进行root,经过一番折腾,发现即使不root也不影响把它变成一款测…

python银行系统模拟演练_python多线程实现代码(模拟银行服务操作流程)

1.模拟银行服务完成程序代码目前,在以银行营业大厅为代表的窗口行业中大量使用排队(叫号)系统,该系统完全模拟了人群排队全过程,通过取票进队、排队等待、叫号服务等功能,代替了人们站队的辛苦。排队叫号软件的具体操作流程为&…

字符串左侧补0_(48)C++面试之最长不含重复字符的子字符串(动态规划)

// 面试题48&#xff1a;最长不含重复字符的子字符串// 题目&#xff1a;请从字符串中找出一个最长的不包含重复字符的子字符串&#xff0c;计算该最长子// 字符串的长度。假设字符串中只包含从a到z的字符。#include <vector> #include <string> #include <iost…

java udp 同一个端口实现收发_Java网络编程之UDP协议

伙伴们注意了&#xff01;小编在这里给大家送上关注福利&#xff1a;搜索微信公众号“速学Java”关注即可领取小编精心准备的资料一份&#xff01;今天我们来聊聊网络编程这部分的内容网络编程1)计算机网络是指将地理位置不同的具有独立功能的多台计算机及其外部设备&#xff0…

java 多态_Java面向对象 —— 多态

前两天已经相继介绍了Java面向对象的三大特性之中的封装、继承&#xff0c;所以今天就介绍Java面向对象的三大特性的最后一项&#xff0c;多态~首先讲一下什么是多态&#xff0c;以及多态需要注意的细节 什么是多态&#xff1a;一个对象具备多种形态&#xff0c;也可以理解为事…

vb6 方法‘ ’作用于对象 失败_JS基础入门-对象的使用

今日背诵小纸条对象是一组属性方法的组合&#xff0c;其中可包含基本值、对象和函数对象的定义1 对象字面量var hero{name: ‘产品小姐姐’&#xff0c;age: 16&#xff0c;weapon: [ ‘头盔’, ‘靴子’, ‘盔甲 ]&#xff0c;sayHi: function ( ) {console.log( this.name ’…

无法从套接字读取更多的数据 oracle_小伙面试时被追问数据库优化,面试前如何埋点反杀?

前言周五的早高峰, 各地软件园地铁站里中出现了不少穿着长袖加绒格子衫, 背双肩电脑包的年轻码农, 现在节气正值 [ 小雪 ] , 11月的全国性突然降温 , 让经历过996摧残的猿们一出地铁站就冻的打了个激灵 , 很庆幸的告诉大家距离放年假还剩不到 37 个工作日, 要买火车票的赶紧预约…

qlineedit限制输入数字_Excel单元格限制录入,实用小技巧

在填写资料表格的时候&#xff0c;为了不防止出错&#xff0c;会在单元格中设置一些技巧&#xff0c;限制对方输入内容&#xff0c;这样可以更好的预防输入错误。那么单元格限制输入技巧是如何实现的呢&#xff1f;1、限制只能录入数字比如单元格是我们要用来填写年龄数据等数字…

java二维数组 内存分配_java中二维数组内存分配

区分三种初始化方式&#xff1a;格式一&#xff1a;数据类型[][] 数组名 new 数据类型[m][n];m:表示这个二维数组有多少个一维数组。n:表示每一个一维数组的元素有多少个。//例&#xff1a;int arr[][]new int[3][2];如下图格式二&#xff1a;数据类型[][] 数组名 new 数据类…

word公式插件_如何快速输入复杂的数学公式?这里有 3 个实用技巧

不管你是不是科研狗&#xff0c;都可能遇到过在文章中插入公式。而我们最常用的就是使用 Word 自带的公式编辑器输入&#xff0c;Word 公式可以很好地匹配文章的格式&#xff0c;自然地插入文中。有时候处理一个公式简单&#xff0c;但如果你要输入大量公式&#xff0c;键盘、鼠…

java观察者模式本质_6.[研磨设计模式笔记]观察者模式

1.定义定义对象间的一种一对多的依赖关系&#xff0c;当一个对象的状态发生改变时&#xff0c;所有依赖于它的对象都得到通知并自动更新。2.解决问题——订阅报纸看起来订阅者是直接根有据打交道&#xff0c;但实际上&#xff0c;订阅者的订阅数据是被邮寄传递到报社&#xff0…

粒子群算法tsp java_粒子群算法解决TSP问题

1. 粒子群算法简介粒子群算法(particle swarm optimization&#xff0c;PSO)由Kennedy和Eberhart在1995年提出&#xff0c;属于进化算法的一种&#xff0c;是通过对模拟鸟群扑食行为设计的。基本思想&#xff1a;从随机解出发&#xff0c;通过迭代寻找最优解&#xff0c;通过适…

hive mysql性能_Hive数据库安全审计功能

【Hive数据库安全审计简介】Hive数据库安全审计是一款基于数据库通讯协议准确分析和SQL完全解析技术的数据库安全审计系统。实现了对数据库操作、访问用户及外部应用用户的审计&#xff0c;可以用于安全合规、用户行为分析、运维监控、风控审计、事件追溯等与数据库安全相关的管…

linux安装sz rz_超级好用的文件传输命令rz与sz

做生物信息经常需要在本地客户端与服务器之间进行文件的传输&#xff0c;例如将要分析的数据传到Linux服务器上&#xff0c;进行分析&#xff0c;分析结束之后将结果下载到本地windows系统进行查看。以前我们都推荐大家使用比较稳定&#xff0c;并且支持断点续传的Filezilla或者…

visual studio 按钮判断管理员和用户_用户管理的设计原则

后台的使用者通常是企业的各类权限的管理员&#xff0c;通过对用户管理的合理设计&#xff0c;让管理员准确、高效地完成既定任务&#xff0c;来实现用户管理的价值。后台产品的使命是为企业提升效率&#xff0c;降低成本&#xff0c;而用户管理是后台的核心部分&#xff0c;用…

windows。forms.timer设置第一次不等待_面试官:换人!他连 TCP 这几个参数都不懂(一)...

前言TCP 性能的提升不仅考察 TCP 的理论知识&#xff0c;还考察了对于操作系统提供的内核参数的理解与应用。TCP 协议是由操作系统实现&#xff0c;所以操作系统提供了不少调节 TCP 的参数。如何正确有效的使用这些参数&#xff0c;来提高 TCP 性能是一个不那么简单事情。我们需…