• Python async模块使用(杂文)


    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


    协程:协程(Coroutine),也可以被称为微线程,是一种用户态内的上下文切换技术。简而言之,其实就是通过一个线程实现代码块相互切换执行

    Python对协程的支持是通过generator实现的。
    在generator中,我们不但可以通过for循环来迭代,还可以不断调用next()函数获取由yield语句返回的下一个值。
    但是Python的yield不但可以返回一个值,它还可以接收调用者发出的参数。

    一、什么是 generator(生成器)?

    在Python中,这种一边循环一边计算的机制,称为生成器:generator。通过给定一个算法然后在调用的过程中计算真实值。
    当需要从generator中获取值的时候可以使用next(),但是一般使用for循环进行获取。

    generator的实现方式

    1. 生成器,使用()表示
      如:[1, 2, 3, 4, 5],生成器方法:
    data = [1, 2, 3, 4, 5]
    (x * x for x in len(data))
    
    • 1
    • 2
    1. 函数定义
      在一些逻辑复杂的场景下,使用第一种方法不太合适,因此存在类型函数定义的方式,如:
    def num(x):
        while (x < 10):
            print(x * x)
            x += 1
    g = num(1)
    for item in g:
        print(item)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    当函数中出现yield的时候,此时就成为generator

    def num(x):
        while (x < 10):
            yield x * x  # 返回结果,下次从这个地方继续?
            x += 1
    g = num(1)  # 返回的是generator对象
    for item in g:
        print(item)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    变成generator的函数,在每次调用next()的时候执行,遇到yield语句返回,再次执行时从上次返回的yield语句处继续执行。

    二、使用asyncio 实现异步io

    异步io通过事件循环和协程函数实现
        事件循环即不断监察内部的任务,如果存在则执行;任务分为可执行和正在执行;由事件循环决定处理任务,如果任务列表为空,事件终止。

    import asyncio
     
    # 生成或获取事件循环对象loop
    loop = asyncio.get_event_loop()  
     
    # 将协程函数(任务)提交到事件循环的任务列表中,协程函数执行完成之后终止。
    # run_until_complete 会检查协程函数的运行状态,并执行协程函数
    loop.run_until_complete( func() ) 
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

        协程函数:由 async def 修饰的函数;相比于普通的def, 如 def func(),可以直接接收到函数返回的值;但是对于协程函数返回的是一个协程对象。
    想要运行协程函数,需要将这个对象交给事件循环进行处理。

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

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

    可等待对象: 协程对象, Future, Task对象 可理解为IO等待
    response : 等待的结果
    await 遇到IO操作会挂起当前协程(任务),当前协程挂起时,事件循环可以去执行其他协程(任务)
    注意:可等待对象若是协程对象则变成串行,若是Task对象则并发运行
    Task对象,可以在事件循环列表中添加多个任务。可以通过**asyncio.create_task(协程对象)**的方式创建Task对象

    import asyncio
    import time, datetime
    
    async def display(num):
        pass
    
    tasks = []
    for num in range(10):
        tasks.append(display(num))  # 生成任务列表
    loop = asyncio.get_event_loop()
    loop.run_until_complete(asyncio.wait(tasks))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    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"])
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • post请求
    1. 异步的执行两个任务
    2. 在网络请求中,一个请求就是一个会话,然后aiohttp使用的是ClientSession来管理会话
    3. 使用session.method发送请求
    4. 对于响应信息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))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    在这里插入图片描述


    继续积累。。。
  • 相关阅读:
    关于vue封装form表单单向流数据问题
    《向经典致敬》第二届粤港澳大湾区著名歌唱家音乐会完美落幕
    无法加载文件xxx,此系统上禁止运行脚本
    【Mycat2实战】四、Mycat实现分库分表【概念篇】
    【Linux】简化自用-Win10安装VMware和CentOS
    Java全栈开发第一阶段--01.Java基础编程(基本语法-进制)
    “加薪”、“洗手间”都不能提?亚马逊内部员工通讯 App 曝光
    流程编排、如此简单-通用流程编排组件JDEasyFlow介绍
    时序分析基础(3)——output_delay
    python mongodb数据查询
  • 原文地址:https://blog.csdn.net/qq_45888932/article/details/125424397