python中的yeild

python中的yeild您可能听说过 带有 yield 的函数在 Python 中被称之为 generator 生成器 何谓 generator 我们先抛开 generator 以一个常见的编程题目来展示 yield 的概念 如何生成斐波那契數列斐波那契 Fibonacci 數列是一个非常简单的递归数列 除第一个和第二个数外 任意一个数都可由前两个数相加得到 用计算机程序输出斐波那契數列的前

生成器是这样一个函数,它记住上一次返回时在函数体中的位置。对生成器函数的第二次(或第 次)调用跳转至该函数中间,而上次调用的所有局部变量都保持不变。

生成器不仅“记住”了它数据状态;生成器还“记住”了它在流控制构造(在命令式编程中,这种构造不只是数据值)中的位置。

生成器的特点

     生成器是一个函数,而且函数的参数都会保留。

     迭代到下一次的调用时,所使用的参数都是第一次所保留下的,即是说,在整个所有函数调用的参数都是第一次所调用时保留的,而不是新创建的

Python中,yield就是这样的一个生成器。

yield 生成器的运行机制

当你问生成器要一个数时,生成器会执行,直至出现 yield 语句,生成器把 

     yield 的参数给你,之后生成器就不会往下继续运行。 当你问他要下一个数时,他会从上次的状态。开始运行,直至出现yield语句,把参数给你,之后停下。如此反复直至退出函数。(以上关于yield的描述,在后面列举一个简单的例子来解释这段话)

yield的使用

python中,当你定义一个函数,使用了yield关键字时,这个函数就是一个生成器,它的执行会和其他普通的函数有很多不同,函数返回的是一个对象,而不是你平常所用return语句那样,能得到结果值。如果想取得值,那得调用next()函数,如:

[python]  view plain   copy

  print ?

  1. c = h() #h()包含了yield关键字  
  2. #返回值  
  3. c.next()  

每当调用一次迭代器的next函数,生成器函数运行到yield之处,返回yield后面的值且在这个地方暂停,所有的状态都会被保持住,直到下次next函数被调用,或者碰到异常循环退出。

下面,来看看以下的例子代码吧,是用来说明yield运行机制的。

[python]  view plain   copy

  print ?

  1. def fib(max):  
  2.     a, b = 11  
  3.     while a < max:  
  4.         yield a #generators return an iterator that returns a stream of values.  
  5.         a, b = b, a+b  

程序运行:

[python]  view plain   copy

  print ?

  1. for n in fib(15):  
  2.     print n  


从前面的运行机制描述中,可以获知,程序运行到yield这行时,就不会继续往下执行。而是返回一个包含当前函数所有参数的状态的iterator对象。目的就是为了第二次被调用时,能够访问到函数所有的参数值都是第一次访问时的值,而不是重新赋值。

程序第一次调用时:

[python]  view plain   copy

  print ?

  1. def fib(max):  
  2.     a, b = 11  
  3.     while a < max:  
  4.         yield a #这时a,b值分别为1,1,当然,程序也在执行到这时,返回  
  5.         a, b = b, a+b  


程序第二次调用时:

从前面可知,第一次调用时,a,b=1,1,那么,我们第二次调用时(其实就是调用第一次返回的iterator对象的next()方法),程序跳到yield语句处,

执行a,b = b, a+b语句,此时值变为:a,b = 1, (1+1) => a,b = 1, 2

程序继续while循环,当然,再一次碰到了yield a 语句,也是像第一次那样,保存函数所有参数的状态,返回一个包含这些参数状态的iterator对象。

等待第三次的调用….

[python]  view plain   copy

  print ?

  1. def fib(max):  
  2.     a, b = 11  
  3.     while a < max:  
  4.         yield a   
  5.         a, b = b, a+b  



通过上面的分析,可以一次类推的展示了yield的详细运行过程了!

通过使用生成器的语法,可以免去写迭代器类的繁琐代码,如,上面的例子使用迭代类来实现,代码如下:

[python]  view plain   copy

  print ?

  1. class Fib:  
  2.     def __init__(self, max):  
  3.         self.max = max  
  4.     def __iter__(self):  
  5.         self.a = 0  
  6.         self.b = 1  
  7.         return self  
  8.     def next(self):  
  9.         fib = self.a  
  10.         if fib > self.max:  
  11.             raise StopIteration  
  12.         self.a, self.b = self.b, self.a + self.b  
  13.         return fib  



yield其他例子展示:排列,组合

#生成全排列    

[python]  view plain   copy

  print ?

  1. def perm(items, n = None):  
  2.     if n is None:  
  3.         n = len(items)  
  4.     for i in range(len(items)):  
  5.         v = items[i:i+1]  
  6.         if n==1:  
  7.             yield v  
  8.         else:  
  9.             rest = items[:i] + items[i+1:]  
  10.             for p in perm(rest, n-1):  
  11.                 yield v + p  
  12. def comb(items, n = None):  
  13.     if n is None:  
  14.         n = len(items)  
  15.     else:  
  16.         for i in range(len(items)):  
  17.             v = items[i:i+1]  
  18.             if 1 == n:  
  19.                 yield v  
  20.             else:  
  21.                 rest = items[i+1:]  
  22.                 for c in comb(rest, n-1):  
  23.                     yield v + c  

另一个 yield 的例子来源于文件读取。如果直接对文件对象调用 read() 方法,会导致不可预测的内存占用。好的方法是利用固定长度的缓冲区来不断读取文件内容。通过 yield,我们不再需要编写读文件的迭代类,就可以轻松实现文件读取:

清单 9. 另一个 yield 的例子
 def read_file(fpath): BLOCK_SIZE = 1024 with open(fpath, 'rb') as f: while True: block = f.read(BLOCK_SIZE) if block: yield block else: return







版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/233686.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • c语言之选择排序法

    c语言之选择排序法c语言之选择排序法啊,这是我第一次写文章,可能会有很多不足,希望大家可以给我指出。问题:选择法排序题目描述输入一个正整数n,再输入n个整数,将他们从大到小排序后输出。样例输入525134样例输出54321//精髓就是要交换数值大小;#include<stdio.h>intmain(){ intn; scanf(“%d”,&n);//定义有多少数; inti,a[n]; for(i=0;i<n;i++) { s

    2022年6月25日
    26
  • mysql econnreset_Nodejs 套接字报错处理 Error: read ECONNRESET

    mysql econnreset_Nodejs 套接字报错处理 Error: read ECONNRESET错误信息:Error:readECONNRESETatTCP.onStreamRead(internal/stream_base_commons.js:162:27)出现上述情况一般是客户端关闭了socket连接导致的错误,这个错误会导致程序的异常退出解决办法:varpReq=http.request(options,function(pRes){cSock.writeHead…

    2022年6月17日
    83
  • PO模式 – 目录结构

    PO模式 – 目录结构前言:我们为什么要用到PO模式?因为随着时间的迁移。测试套件将持续的增长,脚本也将变得越来越多。如果需要维护10个页面,100个页面,甚至1000个呢?那么页面元素的任何改变都会让我们的脚本维护变得繁琐复杂,而且变得耗时易出错那怎么解决呢?ui自动化中,常用的一种方式,引入PageObject(PO):页面对象模式来解决,po能让我们的测试代码变得可读性更好,可维护性高,复用性高。PO是…

    2022年4月30日
    53
  • 欧拉角pitch、yaw,roll的理解_彻底搞懂四元数

    欧拉角pitch、yaw,roll的理解_彻底搞懂四元数目录0、简介一、四元数的定义二、欧拉角到四元数的转换2.1公式:2.2code:三、四元数到欧拉角的转换3.1公式3.2code:3.3四元素到旋转矩阵转换四.奇点五.矢量旋转证明:六.其他参考0、简介四元数与欧拉角之间的转换百度百科四元素在3D图形学中,最常用的旋转表示方法便是四元数和欧拉角,比起矩阵来具……

    2022年9月22日
    4
  • php程序员面试题(偏中级面试题)「建议收藏」

    php程序员面试题(偏中级面试题)

    2022年2月19日
    40
  • 边缘检测算子Canny原理概述并利用OpenCV的库函数Canny()对图像进行边缘检测[通俗易懂]

    边缘检测算子Canny原理概述并利用OpenCV的库函数Canny()对图像进行边缘检测[通俗易懂]图像边缘检测的概念和大概原理可以参考我的另一篇博文,链接如下:https://blog.csdn.net/wenhao_ir/article/details/51743382本篇博文介绍边缘检测算子Canny,并利用OpenCV的库函数Canny()对图像进行边缘检测。Canny算子是JohnCanny在1986年发表的论文中首次提出的边缘检测算子,该算子检测性能比较好,应用广泛。Canny算法被推崇为当今最优的边缘检测的算法。Canny算子进行边缘检测的原理和步骤如下:⑴消除噪声。边缘

    2022年5月29日
    40

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号