Python,从yield到协程

#codomg:UTF-8
#出处:http://www.runoob.com/w3cnote/python-yield-used-analysis.html
"""怎么用Python生成一个Fibonacci数列"""
def Fibonacci_1(N):
""" 斐波那契1.0
        简单地输出N个斐波那契数
        点评:大概是一个程序员new bee写的,用print复用性太差了
"""
    n, a, b = 0, 0, 1
    while n < N:
        print b
        a, b = b, a+b
        n=n+1

def Fibonacci_2(N):
""" 斐波那契2.0
        返回到N的斐波那契数组
        点评:应该是一个稍有经验的程序员的成果,但是内存随着N增大而增大,浪费内存
"""
    n, a, b = 0, 0, 1
    L=[]
    while n < N:
        L.append(b)
        a, b = b, a+b
        n = n+1
    return L

class Fibonacci_3(object)):
""" 斐波那契3.0
        用一个迭代对象来存斐波那契数
        点评:看样子是出自一个经验丰富的程序员之手,但是我只是要一个斐波那契数,这么长的代码吗?
"""
    def __init__(self, N):
        self.N = N
        self.n, self.a, self.b = 0, 0, 1
    
    def __iter__(self):
        return self
    
    def next(self):
        if self.n < self.N:
            r = self.b
            self.a, self.b = self.b, self.a+self.b
            self.n = self.n + 1
            return r
        raise StopIteration()

def Fibonacci_4(N):
""" 斐波那契4.0
        yield迭代
        厉害了,只跟1.0差一行代码,这一定是一位资深大牛的作品,其实只是熟悉迭代,就是这么简单
"""
    n, a, b = 0, 0, 1
    while n < max:
        yield b
        a, b = b, a+b
        n = n+1

#下面的意见部分参考了知乎用户饭后温柔的意见
#https://www.zhihu.com/question/20511233/answer/24459911
"""迭代,就是协程的典型应用
    一般来说,函数我们输入一样的值,就会产生相同的输出。
    而对于迭代对象,每次调用都仅执行一次,但保留当前状态,等待下次调用,衔接上次执行
    达到了相同的输入,不同的输出
    类似于存在一个限制了使用域的全局变量
    虽然输入都一样,但是每次调用都会变更这个变量
    yield就是触发协程的一个关键字
    yield会保存当前内存上下文状态,然后return当前值
    下次调用函数读取状态继续执行
    在上古时代,C语言还没一统天下的那个年代
    函数的命名五花八门,其中有一个名字其实特别形象,是真正符合函数与系统定义的
    那就是“过程”
    对于CPU与系统来说。
    过程是一个独立的,单向的,不可中断的操作,一个过程就要进行到底
    协程是一个独立的,单向的,可中断的操作,多个协程可切换操作,在早期单线程处理器时代可以达到最大化利用CPU的目的,即使现在也是一样
    线程是一组协程或过程组成的活动,多个线程共享资源,但是对资源要进行控制,是多线程处理器的最小处理单元
    进程是一组线程组成的活动,具有较高的独立性,是系统进行资源分配和调度的基本单位
    看到这里我忽然想,一组进程归并起来是不是能出来一个新的程
    这样就叫他集程好了
    其实就是服务器集群
    k8s就是一个集程管理系统,用于服务器集群的负载均衡,也就是资源分配和调度
    厉害了,一个新的概念就此诞生。
"""

""" 如果还不是很清晰协程的概念,再来廖雪峰提供的例子 """
#出处:https://www.liaoxuefeng.com/wiki/001374738125095c955c1e6d8bb493182103fac9270762a000/0013868328689835ecd883d910145dfa8227b539725e5ed000
import time

def consumer():
    r = ''
    while True:
        n = yield r
        if not n:
            return
        print('[CONSUMER] Consuming %s...' % n)
        time.sleep(1)
        r = '200 OK'

def produce(c):
    c.next()
    n = 0
    while n < 5:
        n = n + 1
        print('[PRODUCER] Producing %s...' % n)
        r = c.send(n)
        print('[PRODUCER] Consumer return: %s' % r)
    c.close()

if __name__=='__main__':
    c = consumer()
    produce(c)

"""执行结果
    [PRODUCER] Producing 1...
    [CONSUMER] Consuming 1...
    [PRODUCER] Consumer return: 200 OK
    [PRODUCER] Producing 2...
    [CONSUMER] Consuming 2...
    [PRODUCER] Consumer return: 200 OK
    [PRODUCER] Producing 3...
    [CONSUMER] Consuming 3...
    [PRODUCER] Consumer return: 200 OK
    [PRODUCER] Producing 4...
    [CONSUMER] Consuming 4...
    [PRODUCER] Consumer return: 200 OK
    [PRODUCER] Producing 5...
    [CONSUMER] Consuming 5...
    [PRODUCER] Consumer return: 200 OK

    注意到因为使用的yield,consumer变成一个generator(迭代生成器),把一个consumer传入produce后:

    首先调用c.next()启动迭代;

    然后,一旦生产了东西,通过c.send(n)切换到consumer执行;

    consumer通过yield拿到消息,处理,又通过yield把结果传回;

    produce拿到consumer处理的结果,继续生产下一条消息;

    produce决定不生产了,通过c.close()关闭consumer,整个过程结束。

    整个流程无锁,由一个线程执行,produce和consumer协作完成任务,所以称为“协程”,而非线程的抢占式多任务。

    最后套用Donald Knuth的一句话总结协程的特点:

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

推荐阅读更多精彩内容