异步操作之asyncio(一)

1.简介

1)并发、并行、同步、异步、多线程的区别?

并发:同一时间段有几个程序都处于已经启动到运行完毕之间,并且这几个程序都在同一个处理机上运行,并发的两种关系是同步和互斥;

互斥:进程之间访问临界资源时相互排斥的现象;

同步:进程之间存在依赖关系,一个进程结束的输出作为另一个进程的输入。具有同步关系的一组并发进程之间发送的信息称为消息或者事件;

并行:单处理器中进程被交替执行,表现出一种并发的外部特征;在多处理器中,进程可以交替执行,还能重叠执行,实现并行处理,并行就是同事发生的多个并发事件,具有并发的含义,但并发不一定是并行,也就是说事件之间不一定要同一时刻发生;

多线程:多线程是进程中并发运行的一段代码,能够实现线程之间的切换执行;

异步:和同步相对,同步是顺序执行,而异步是彼此独立,在等待某个事件的过程中继续做自己的事,不要等待这一事件完成后再工作。线程是实现异步的一个方式,异步是让调用方法的主线程不需要同步等待另一个线程的完成,从而让主线程干其他事情。

异步和多线程:不是同等关系,异步是目的,多线程只是实现异步的一个手段,实现异步可以采用多线程技术或者交给其他进程来处理。

2)同步和异步,阻塞和非阻塞

同步和异步关注的是结果消息的通信机制

- 同步:同步的意思就是调用方需要主动等待结果的返回

- 异步:异步的意思就是不需要主动等待结果的返回,而是通过其他手段比如,状态通知,回调函数等。

阻塞和非阻塞主要关注的是等待结果返回调用方的状态

阻塞:是指结果返回之前,当前线程被挂起,不做任何事

非阻塞:是指结果在返回之前,线程可以做一些其他事,不会被挂起。

---------------------

异步非阻塞:异步非阻塞这也是现在高并发编程的一个核心

2.asyncio简介

asyncio模块提供了使用协程构建并发应用的工具。它使用一种单线程单进程的的方式实现并发,应用的各个部分彼此合作, 可以显示的切换任务,一般会在程序阻塞I/O操作的时候发生上下文切换如等待读写文件,或者请求网络。同时asyncio也支持调度代码在将来的某个特定事件运行,从而支持一个协程等待另一个协程完成,以处理系统信号和识别其他一些事件。

1)异步并发的概念

对于其他的并发模型大多数采取的都是线性的方式编写。并且依赖于语言运行时系统或操作系统的底层线程或进程来适当地改变上下文,而基于asyncio的应用要求应用代码显示的处理上下文切换。asyncio提供的框架以事件循环(event loop)为中心,程序开启一个无限的循环,程序会把一些函数注册到事件循环上。当满足事件发生的时候,调用相应的协程函数。

事件循环

事件循环是一种处理多并发量的有效方式,在维基百科中它被描述为「一种等待程序分配事件或消息的编程架构」,我们可以定义事件循环来简化使用轮询方法来监控事件,通俗的说法就是「当A发生时,执行B」。事件循环利用poller对象,使得程序员不用控制任务的添加、删除和事件的控制。事件循环使用回调方法来知道事件的发生。它是asyncio提供的「中央处理设备」,支持如下操作:

    (1) 注册、执行和取消延迟调用(超时)

    (2) 创建可用于多种类型的通信的服务端和客户端的Transports

    (3) 启动进程以及相关的和外部通信程序的Transports

    (4) 将耗时函数调用委托给一个线程池

    (5) # 单线程(进程)的架构也避免的多线程(进程)修改可变状态的锁的问题。


2) 与事件循环交互的应用要显示地注册将运行的代码,让事件循环在资源可用时向应用代码发出必要的调用。

Future

future是一个数据结构,表示还未完成的工作结果。事件循环可以监视Future对象是否完成。从而允许应用的一部分等待另一部分完成一些工作。

Task

task是Future的一个子类,它知道如何包装和管理一个协程的执行。任务所需的资源可用时,事件循环会调度任务允许,并生成一个结果,从而可以由其他协程消费。



3)一个异步方法:

async def async_double(x):

        return 2 * x

print(await async_double(x))

从外观上看异步方法和标准方法没什么区别只是前面多了个async。

"Async" 是"asynchronous"的简写,为了区别于异步函数,我们称标准函数为同步函数,

从用户角度异步函数和同步函数有以下区别:

要调用异步函数,必须使用await关键字。 因此,不要写regulardouble(3),而是写await asyncdouble(3).

不能在同步函数里使用await,否则会出错。

-------------------------------------------------------------------------------------------------------------------

4)启动一个协程

一般异步方法被称之为协程(Coroutine)。asyncio事件循环可以通过多种不同的方法启动一个协程。一般对于入口函数,最简答的方法就是使用run_until_complete,并将协程直接传入这个方法。

import asyncio

async def foo:

    print("这是一个协程")

    return 21

if __name__ == '__main__':

    loop = asyncio.get_event_loop()

    try:

        print("开始运行协程")

         coro = foo

         print("进入事件循环")

        # loop.run_until_complete(coro())

        result =  loop.run_until_complete(coro())

        print(f"run_until_complete可以获取协程的{result},默认输出None")

     finally:

         print("关闭事件循环")

         loop.close

第一步首先得到一个事件循环的应用也就是定义的对象loop。可以使用默认的事件循环,也可以实例化一个特定的循环类(比如uvloop),这里使用了默认循环run_unti_lcomplete(coro())方法用这个协程启动循环,协程返回时这个方法将停止循环。

rununtilcomplete的参数是一个futrue对象。当传入一个协程,其内部会自动封装成task,其中task是Future的子类。关于task和future后面会提到



#协程调用协程
import asyncio

async def result1():

    print("这是result1协程")

    return "result1"

async def result2(arg):

    print("这是result2协程")

    return f"result2接收了一个参数,{arg}"

async def main():

    print("主协程")

    print("等待result1协程运行")

    res1 = await result1()

    print("等待result2协程运行")

    res2 = await result2(res1)

    return (res1,res2)

if __name__ == '__main__':

    loop = asyncio.get_event_loop()

    try:

        result = loop.run_until_complete(main())

        print(f"获取返回值:{result}")

    finally:

        print("关闭事件循环")

        loop.close



5) 协程中调用普通函数

在协程中可以通过一些方法去调用普通的函数。可以使用的关键字有callsoon,calllater,call_at。

(1)call_soon

可以通过字面意思理解调用立即返回(或者说,立即调用回调函数)。

loop.call_soon(callback, *args, context=None)

在下一个迭代的时间循环中立刻调用回调函数,大部分的回调函数支持位置参数,而不支持"关键字参数",如果是想要使用关键字参数,则推荐使用functools.aprtial对方法进一步包装.可选关键字context允许指定要运行的回调的自定义contextvars.Context。当没有提供上下文时使用当前上下文。在Python 3.7中, asyncio协程加入了对上下文的支持。使用上下文就可以在一些场景下隐式地传递变量,比如数据库连接session等,而不需要在所有方法调用显示地传递这些变量。

import asyncio

import functools

def callback(args, kwargs="defalut"):

    # f支持字符串里加入变量,相当于动态更新

    print(f"普通函数做为回调函数,获取参数:{args},{kwargs}")

async def main(loop):

    print("注册callback")

    loop.call_soon(callback, 1)

    wrapped= functools.partial(callback,kwargs="not defalut") # 将关键字参数先行封装

    loop.call_soon(wrapped, 2)

    await asyncio.sleep(0.2)

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        print(loop,'\n',type(loop))

        loop.run_until_complete(main(loop))

    finally:

        loop.close()


(2)call_later

loop.call_later(delay, callback, *args, context=None)

首先简单的说一下它的含义,就是事件循环在delay多长时间之后才执行callback函数

import asyncio

def callback(n):

    print(f"callback {n} invoked")

async def main(loop):

    print("注册callbacks")

    loop.call_later(0.2, callback, 1)

    loop.call_later(0.1, callback, 2)

    loop.call_soon(callback, 3)

    await asyncio.sleep(0.4)

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        loop.run_until_complete(main(loop))

    finally:

        loop.close()

1.callsoon会在calllater之前执行,和它的位置在哪无关

2.call_later的第一个参数越小,越先执行。


(3) call_at()

loop.call_at(when, callback, *args, context=None)

callat第一个参数的含义代表的是一个单调时间,它和我们平时说的系统时间有点差异,这里的时间指的是事件循环内部时间,可以通过loop.time获取,然后可以在此基础上进行操作.实际上calllater内部就是调用的call_at。

import asyncio

def call_back(n, loop):

    print(f"callback {n} 运行时间点{loop.time()}")

async def main(loop):

    now= loop.time()

    print("当前的内部时间", now)

    print("循环时间", now)

    print("注册callback")

    loop.call_at(now+ 0.1, call_back, 1, loop)

    loop.call_at(now+ 0.2, call_back, 2, loop)

    loop.call_soon(call_back, 3, loop)

    await asyncio.sleep(1)

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        print("进入事件循环")

        loop.run_until_complete(main(loop))

    finally:

        print("关闭循环")

        loop.close()



6)future

future表示还没有完成的工作结果。事件循环可以通过监视一个future对象的状态来指示它已经完成。future对象有几个状态:

import asyncio

def foo(future, result):

    print(f"此时future的状态:{future}")

    print(f"设置future的结果:{result}")

    future.set_result(result)

    print(f"此时future的状态:{future}")

    print(f"此时future的状态:{future.result()}") # future.result(),需要先设置

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        all_done= asyncio.Future()

        loop.call_soon(foo, all_done, "Future is done!")

        print("进入事件循环")

        result= loop.run_until_complete(all_done)

        print("返回结果", result)

    finally:

        print("关闭事件循环")

        loop.close()

        print("获取future的结果", all_done.result())

---------------------------------------------------------------------------------

future和协程一样可以使用await关键字获取其结果。

def foo(future, result):

    print("设置结果到future", result)

    future.set_result(result)

async def main(loop):

    all_done= asyncio.Future()

    print("调用函数获取future对象")

    loop.call_soon(foo, all_done, "the result")

    result= await all_done# 获取状态

    # result = all_done.result() 没有设置,不能用

    print("获取future里的结果", result)

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        loop.run_until_complete(main(loop))

    finally:

        loop.close()

Future回调

Future 在完成的时候可以执行一些回调函数,回调函数按注册时的顺序进行调用:

import asyncio

import functools,time

def callback(future, n):

    print(future)

    print('{}: future done: {}'.format(n, future.result()))

async def register_callbacks(all_done):

    print('注册callback到future对象')

    # all_done是future对象,最为第一个参数,填入,注册完,并不会立即回掉

    all_done.add_done_callback(functools.partial(callback, n=1))

    all_done.add_done_callback(functools.partial(callback, n=2))

async def main(all_done):

    await register_callbacks(all_done)

    print('设置future的结果,该协程结束后执行回调')

    all_done.set_result('the result')

    time.sleep(2)

    print('===========')

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        all_done= asyncio.Future()

        loop.run_until_complete(main(all_done))

    finally:

        loop.close()

通过adddonecallback方法给futrue任务添加回调函数,当future执行完成的时候,就会调用回调函数。


三、并发的执行任务

 任务(Task)是与事件循环交互的主要途径之一。任务可以包装协程,可以跟踪协程何时完成。任务是Future的子类,所以使用方法和future一样。协程可以等待任务,每个任务都有一个结果,在它完成之后可以获取这个结果。因为协程是没有状态的,我们通过使用create_task方法可以将协程包装成有状态的任务。还可以在任务运行的过程中取消任务。

import asyncio

async def child():

    print("进入子协程")

    return "the result"

async def main(loop):

    print("将协程child包装成任务")

    task= loop.create_task(child())

    # print("通过cancel方法可以取消任务")

# task.cancel()

# print(9999)

    try:

        await task

# print(1000)

    except asyncio.CancelledError:

        print("取消任务抛出CancelledError异常")

    else:

        print("获取任务的结果", task.result())

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        loop.run_until_complete(main(loop))

    finally:

        loop.close()

另外出了使用loop.create_task()将协程包装为任务外还可以使用asyncio.ensurefuture(coroutine)建一个task。在python3.7中可以使用asyncio.create_task创建任务。


四、组合协程

一系列的协程可以通过await链式的调用,但是有的时候我们需要在一个协程里等待多个协程,比如我们在一个协程里等待1000个异步网络请求,对于访问次序有没有要求的时候,就可以使用另外的关键字wait或gather来解决了。

1) wait可以暂停一个协程,直到后台操作完成。

等待多个协程:

import asyncio

async def num(n):

    try:

        await asyncio.sleep(n*0.1)

        return n

except asyncio.CancelledError:

        print(f"数字{n}被取消")

        raise

async def main():

    tasks= [num(i) for iin range(10)]

    complete, pending= await asyncio.wait(tasks, timeout=0.5)

    for iin complete:

        print("当前数字",i.result())

    if pending:

        print("取消未完成的任务")

        for pin pending:

            p.cancel()

2) gather的使用

gather的作用和wait类似不同的是。

1.gather任务无法取消。

2.返回值是一个结果列表

3.可以按照传入参数的顺序,顺序输出。

complete= await asyncio.gather(*tasks)

for iin complete:

    print("当前数字", i)

gather通常被用来阶段性的一个操作,做完第一步才能做第二步:

"""

可以通过上面结果得到如下结论:1.step1和step2是并行运行的。

2.gather会等待最耗时的那个完成之后才返回结果,耗时总时间取决于其中任务最长时间的那个。"""

async def step1(n, start):

    await asyncio.sleep(n)

    print("第一阶段完成")

    print("此时用时", time.time() - start)

    return n

async def step2(n, start):

    await asyncio.sleep(n)

    print("第二阶段完成")

    print("此时用时", time.time() - start)

    return n

async def main():

    now= time.time()

    result= await asyncio.gather(step1(5, now), step2(2, now))

    for iin result:

        print(i)

        print("总用时", time.time() - now)

if __name__== '__main__':

    loop= asyncio.get_event_loop()

    try:

        loop.run_until_complete(main())

    finally:

        loop.close()

3)任务完成时进行处理

"""

任务完成时进行处理ascomplete是一个生成器,会管理指定的一个任务列表,并生成他们的结果。每个协程结束运行时一次生成一个结果。

与wait一样,ascomplete不能保证顺序,不过执行其他动作之前没有必要等待所以后台操作完成。"""

import asyncio

import time

async def foo(n):

    # time.sleep(2) # 暂停上下文全局时间

    print('Waiting: ', n)

    await asyncio.sleep(n)

    return n

async def main():

    coroutine1= foo(1)

    coroutine2= foo(2)

    coroutine3= foo(4)

    tasks= [

    asyncio.ensure_future(coroutine1),

    asyncio.ensure_future(coroutine2),

    asyncio.ensure_future(coroutine3)

    ]

    for taskin asyncio.as_completed(tasks):

        result= await task

print('Task ret: {}'.format(result))

start= time.time()

loop= asyncio.get_event_loop()

done= loop.run_until_complete(main())

print(time.time() - start)

可以发现结果逐个输出。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容