2024年最新【秋招面试专题解析】,夜来风雨声,Python协程知多少(2),2024年最新面试文件

收集整理了一份《2024年最新Python全套学习资料》免费送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img



既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Python知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来

如果你需要这些资料,可以添加V无偿获取:hxbc188 (备注666)
img

正文

next(my_coro)

-> coroutine started

my_coro.send(42)

-> coroutine received: 42

Traceback (most recent call last):

File “”, line 1, in

StopIteration

之所以yield可以放右边,是因为协程可以接收调用方使用​​.send()​​推送的值。

yield放在右边以后,它的右边还能再放个表达式,请看下面这个例子:

def simple_coro2(a):

b = yield a

c = yield a + b

my_coro2 = simple_coro2(14)

next(my_coro2)

my_coro2.send(28)

my_coro2.send(99)

执行过程是:

调用next(my_coro2),执行yield a,产出14。

调用my_coro2.send(28),把28赋值给b,然后执行yield a + b,产出42。

调用my_coro2.send(99),把99赋值给c,协程终止。

由此得出结论,对于b = yield a这行代码来说,= 右边的代码在赋值之前执行。

在示例中,需要先调用​​next(my_coro)​​启动生成器,让程序在yield语句处暂停,然后才可以发送数据。这是因为协程有四种状态:

  • ‘GEN_CREATED’ 等待开始执行

  • ‘GEN_RUNNING’ 解释器正在执行

  • ‘GEN_SUSPENDED’ 在yield表达式处暂停

  • ‘GEN_CLOSED’ 执行结束

只有在GEN_SUSPENDED状态才能发送数据,提前做的这一步叫做预激,既可以调用​​next(my_coro)​​​预激,也可以调用​​my_coro.send(None)​​预激,效果一样。

预激协程


协程必须预激才能使用,也就是send前,先调用next,让协程处于GEN_SUSPENDED状态。但是这件事经常会忘记。为了避免忘记,可以定义一个预激装饰器,比如:

from functools import wraps

def coroutine(func):

@wraps(func)

def primer(*args, **kwargs):

gen = func(*args, **kwargs)

next(gen)

return gen

return primer

但实际上Python给出了一个更优雅的方式,叫做yield from,它会自动预激协程。

自定义预激装饰器和yield from是不兼容的。

yield from


yield from相当于其他语言中的await关键字,作用是:在生成器gen中使用yield from subgen()时,subgen会获得控制权,把产出的值传给gen的调用方,即调用方可以直接控制subgen。与此同时,gen会阻塞,等待subgen终止。

yield from可以用来简化for循环中的yield:

for c in “AB”:

yield c

yield from “AB”

yield from x表达式对x做的第一件事就是,调用iter(x),从中获取迭代器。

但yield from的作用远不止于此,它更重要的作用是打开双向通道。如下图所示:

Python协程知多少_main函数

这个图信息量很大,很难理解。

首先要理解这3个概念:调用方、委派生成器、子生成器。

  • 调用方

the client code, a.k.a. the caller def main(data): # <8> results = {} for key, values in data.items(): group = grouper(results, key) # <9> next(group) # <10> for value in values: group.send(value) # <11> group.send(None) # important! <12> # print(results) # uncomment to debug report(results) ​

  • 委派生成器

the delegating generator def grouper(results, key): # <5> while True: # <6> results[key] = yield from averager() # <7> ​

这比术语看着舒服多了。

然后是5条线:send、yield、throw、StopIteration、close。

  • send

协程在yield from表达式处暂停时,main函数可以通过yield from表达式把数据发给yield from语句右边跟着的子协程。

  • yield

yield from语句右边跟着的子协程再把产出的值通过yield from表达式发给main函数。

  • hrow

main函数通过​​group.send(None)​​,传入一个None值,让yield from语句右边跟着的子协程的while循环终止,这样控制权才会交回协程,才能继续执行,否则会一直暂在yield from语句暂停。

  • StopIteration

yield from语句右边跟着的生成器函数返回之后,解释器会抛出StopIteration异常。并把返回值附加到异常对象上,此时协程会恢复。

  • close

main函数执行完以后,会调用close()方法退出协程。

大体流程搞清楚了,更多的技术细节就不继续研究了,有时间的话,在以后的Python原理系列中再学习吧。

yield from经常与Python3.4标准库里的​​@asyncio.coroutine​​装饰器结合使用。

协程用作累加器


这是协程的常见用途,代码如下:

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

如果你需要这些资料,可以添加V无偿获取:hxbc188 (备注666)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
,代码如下:

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

如果你需要这些资料,可以添加V无偿获取:hxbc188 (备注666)
[外链图片转存中…(img-geg42Xfe-1713853484768)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值