PHP前端开发

Python async模块如何使用

百变鹏仔 1个月前 (01-21) #Python
文章标签 如何使用

协程,又称微线程,是一种在用户态进行上下文切换的技术。简而言之,其实就是通过一个线程实现代码块相互切换执行

Python对协程的支持是通过generator实现的。

在generator中,我们不但可以通过for循环来迭代,还可以不断调用next()函数获取由yield语句返回的下一个值。Python的yield不仅可用于返回值,还能接收调用者传递的参数。

一、什么是 generator(生成器)

Python中称为生成器(generator)的机制是一边循环一边计算的。通过给定一个算法然后在调用的过程中计算真实值。

当需要从generator中获取值的时候可以使用next(),但是一般使用for循环进行获取。

立即学习“Python免费学习笔记(深入)”;

generator的实现方式 生成器,使用()表示

如:[1, 2, 3, 4, 5],生成器方法:

data = [1, 2, 3, 4, 5](x * x for x in len(data))

函数定义在一些逻辑复杂的场景下,使用第一种方法不太合适,因此存在类型函数定义的方式,如:

def num(x):    while (x <p>当函数中出现yield的时候,此时就成为generator</p><pre class="brush:py;">def num(x):    while (x <p>变成generator的函数,在每次调用next()的时候执行,遇到yield语句返回,再次执行时从上次返回的yield语句处继续执行。</p><h3>二、使用asyncio 实现异步io</h3><p>异步io通过事件循环和协程函数实现</p><p>事件循环即不断监察内部的任务,如果存在则执行;任务分为可执行和正在执行;由事件循环决定处理任务,如果任务列表为空,事件终止。</p><pre class="brush:py;">import asyncio# 生成或获取事件循环对象loop;类比Java的Netty,我理解为开启一个selectorloop = asyncio.get_event_loop()  # 将协程函数(任务)提交到事件循环的任务列表中,协程函数执行完成之后终止。# run_until_complete 会检查协程函数的运行状态,并执行协程函数loop.run_until_complete( func() )

test demo

import asyncioimport timeasync def test():    print("io等待")    await asyncio.sleep(1)    return 'hello'async def hello():    print("Hello world")    r = await test()    print("hello again")loop = asyncio.get_event_loop()tasks = [hello(), hello()]loop.run_until_complete(asyncio.wait(tasks))loop.close()

协程函数:由 async def 修饰的函数;相比于普通的def, 如 def func(),可以直接接收到函数返回的值;但是对于协程函数返回的是一个协程对象。

想要运行协程函数,需要将这个对象交给事件循环进行处理。

# 测试协程import asyncioimport time, datetime# 异步函数不同于普通函数,调用普通函数会得到返回值# 而调用异步函数会得到一个协程对象。我们需要将协程对象放到一个事件循环中才能达到与其他协程对象协作的效果# 因为事件循环会负责处理子程 序切换的操作。async def Print():    return "hello"loop = asyncio.get_event_loop()loop.run_until_complete(Print)

await:

用法:reponse = await + 可等待对象

可等待对象: 协程对象, Future, Task对象 可理解为IO等待

response : 等待的结果 await 遇到IO操作会挂起当前协程(任务),当前协程挂起时,事件循环可以去执行其他协程(任务) 注意:可等待对象若是协程对象则变成串行,若是Task对象则并发运行 Task对象,可以在事件循环列表中添加多个任务。可以使用`asyncio.create_task()`来创建一个`Task`对象,传递的参数为协程对象

import asyncioimport time, datetimeasync def display(num):    passtasks = []for num in range(10):    tasks.append(display(num))  # 生成任务列表loop = asyncio.get_event_loop()loop.run_until_complete(asyncio.wait(tasks))

asnyc和await是新语法,旧版本为:@asyncio.coroutine 和 yield from

三、aiohttp

asyncio可以实现单线程并发IO操作。如果仅用在客户端,发挥的威力不大。如果把asyncio用在服务器端,例如Web服务器,由于HTTP连接就是IO操作,因此可以用单线程+coroutine实现多用户的高并发支持。

aiohttp则是基于asyncio实现的HTTP框架。

可以类似requests发送请求 get请求

可以通过params参数来指定要传递的参数

async def fetch(session):    async with session.get("http://localhost:10056/test/") as response:        data = json.loads(await response.text())        print(data["data"])

post请求

  • 异步的执行两个任务

  • 在网络请求中,一个请求就是一个会话,然后aiohttp使用的是ClientSession来管理会话

  • 使用session.method发送请求

  • 对于响应信息response, 通过status来获取响应状态码,text()来获取到响应内容;可以在text()指定编码格式。在等待响应结果前,需要在response.text()前添加await关键字

async def init(num):    async with aiohttp.ClientSession() as session:        if num == 1:            time.sleep(5)        print("session begin", num)        async with session.post("http://localhost:10056/hello/", data=json.dumps({"data": "hello"})) as response:            print("client begin", num)            data = json.loads(await response.text())            print(data["data"])        print("session end", num)    print("other")if __name__ == '__main__':    loop = asyncio.get_event_loop()    tasks = [init(1), init(2)]    loop.run_until_complete(asyncio.wait(tasks))