#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的一句话总结协程的特点:
“子程序就是协程的一种特例。”
"""
网友评论