Python之函数进阶-生成器函数

Python之函数进阶-生成器函数

生成器函数

  • Python中有2种方式构造生成器对象:
    • 生成器表达式
    • 生成器函数
      • 函数体代码中包含yield语句的函数
      • 与普通函数调用不同,生成器函数调用返回的是生成器对象

普通函数调用,函数会立即执行直到执行完毕。
生成器函数调用,并不会立即执行函数体,而是返回一个生成器对象,需要使用next函数来驱动这个生 成器对象,或者使用循环来驱动。
生成器表达式和生成器函数都可以得到生成器对象,只不过生成器函数可以写更加复杂的逻辑。

  • 函数返回
    • return语句依然可以终止函数运行,但return语句的返回值不能被获取到
    • return会导致当前函数返回,无法继续执行,也无法继续获取下一个值,抛出StopIteration 异常
    • 如果函数没有显式的return语句,如果生成器函数执行到结尾(相当于执行了return None),一样会抛出StopIteration异常

生成器函数

  • 包含yield语句的生成器函数调用后,生成生成器对象的时候,生成器函数的函数体不会立即执行
  • next(generator) 会从函数的当前位置向后执行到之后碰到的第一个yield语句,会弹出值,并暂停 函数执行
  • 再次调用next函数,和上一条一样的处理过程
  • 继续调用next函数,生成器函数如果结束执行了(显式或隐式调用了return语句),会抛出 StopIteration异常

int('a', 16)

#具体解释如下:
#int() 是Python内置的数值类型转换函数,用于将其他类型的数据转换为整数。
#'a' 是一个十六进制字符串,表示要进行转换的值。
#16 是指定了输入字符串的进制数,这里表示十六进制。
#因此,int('a', 16) 的作用是将十六进制字符串 'a' 转换为对应的十进制整数。在十六进制中,字符 'a' 
#对应的十进制值为 10。所以,执行该代码后,会返回结果 10。

# 返回结果:10
x = ['a', 1, 'b', 20, 'c', 32]
print(sorted(x, key=str))

# 定义一个列表,打印列表的值,转换成str格式进行排序
# 返回结果:[1, 20, 32, 'a', 'b', 'c']
# 定义一个列表x,包含字符串和整数
x = ['a', 1, 'b', 20, 'c', 32]
def fn(x):	# 定义一个函数fn,用于处理列表中的元素
    if isinstance(x, int):	# 如果元素是整数,直接返回该整数
        return x
    else:
        return int(x, 16)	# 如果元素是字符串,尝试将其转换为十六进制整数并返回

print(sorted(x, key=fn))	# 使用sorted函数对列表x进行排序,key参数指定排序依据为fn函数的返回值

# 这样,列表中的字符串元素会按照其对应的十六进制整数值进行排序
# 返回结果:[1, 'a', 'b', 'c', 20, 32]
x = ['a', 1, 'b', 20, 'c', 32]
def fn(x):
    return x if isinstance(x, int) else int(x, 16) # 三元表达式写法

print(sorted(x, key=fn))

# 返回结果:[1, 'a', 'b', 'c', 20, 32]
x = ['a', 1, 'b', 20, 'c', 32]

print(sorted(x, key=lambda x: x if isinstance(x, int) else int(x, 16)))

# lambda表达式
# 返回结果:[1, 'a', 'b', 'c', 20, 32]

生成器函数总结

  • 生成器对象
    • 生成器表达式, 每一次生成器表达式执行一次都会得到一个全新的生成器对象
    • 生成器函数,每一次函数调用都会得到全新的生成器对象。只要有yield语句的函数都是生成器函数
      • 生成器函数每一次执行到yield这一句,把yield的值返回
      • return非常强势,见到它函数就完了。如果你用next驱动生成器对象,碰到了return,
      • 用for不会抛出异常碰到return就退出了

惰性本身是推荐的,还有它用在了协程中,线程、进程之后

# 示例
def inc():
    count = 0
    while True:
        count += 1
        return count
    
# 不会形成死循环,因为有return
inc
    
# inc函数名称
# 返回结果:<function __main__.inc()>
inc()
    
# 调用inc函数返回结果
# 返回结果:1
def inc():
    count = 0
    while True:
        count += 1
        yield count
    
# 创建一个生成器函数
inc
    
# 查看函数名称
# 返回结果:<function __main__.inc()>
inc()
    
# 直接调用函数返回生成器地址
# 返回结果:<generator object inc at 0x1082e9b40>
def inc():
    for i in range(5):
        yield i + 1
    
# 创建生成器函数range(5)
inc, inc()

# 查看inc,调用inc返回一个生成器函数
# 返回结果:(<function __main__.inc()>, <generator object inc at 0x106164040>)
print(1, next(inc()))
print(2, next(inc()))

# print next inc函数,每次都会生成一个全新的函数,所以2次执行得到的结果都是1
# 返回结果:1 1
# 返回结果:2 1
g = inc()
print(next(g))
print(next(g))
print('----------')
for x in g:
    print(x)
print('==========')
print(next(g))

# 创建g变量,值是inc()函数,对变量进行next迭代的就是同一个函数,迭代器使用完之后,在使用next会抛出StopIteration(停止迭代)异常
g = inc()
print(next(g))
print(next(g))
print('----------')
for x in g:
    print(x)
print('==========')
print(next(g))

# 创建g变量,值是inc()函数,对变量进行next迭代的就是同一个函数,迭代器使用完之后,在使用next会抛出StopIteration(停止迭代)异常

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述在这里插入图片描述


无限容器

def foo():     
    count = 0
    while True:
        count += 1
        yield count

# 可以做id生成器
x = foo()
print(1, next(x))
print(2, next(x))
print(3, next(x))

# 生成器对象是惰性对象,你要一个我给你一个,上面的例子是无线容器,容量是无限的
# 返回结果:1 1
# 返回结果:2 2
# 返回结果:3 3

计数器

def inc():
    def foo():
        count = 0
        while True:
            count += 1
            yield count 
            
    c = foo() # 生成器对象
    return c

x = inc() # x的值是inc函数的调用
print(next(x))
print(next(x))
print(next(x))

# 返回结果:1
# 返回结果:2
# 返回结果:3
def inc():
    def foo(): 
        count = 0
        while True:
            count += 1
            yield count
            
    c = foo() # 生成器对象
    def fn():
        return next(c)
    return fn

x = inc()
print(x)
print(x())
print(x())
print(x())

# 返回结果:<function inc.<locals>.fn at 0x1085ee8e0>
# 返回结果:1
# 返回结果:2
# 返回结果:3
def inc():
    def foo(): 
        count = 0
        while True:
            count += 1
            yield count
            
    c = foo() # 生成器对象
    return lambda : next(c) # lambda方式, 闭包是c

x = inc()
print(x)
print(x())
print(x())
print(x())

# 返回结果:<function inc.<locals>.<lambda> at 0x1085edc60>
# 返回结果:1
# 返回结果:2
# 返回结果:3

生成器写斐波那契数列

# fib1:1 fib2:1 fib3:2

# 定义一个生成器函数fib,用于计算斐波那契数列
def fib():
    a = 1	# 初始化第一个斐波那契数a为1
    yield a	# 使用yield关键字返回第一个斐波那契数a
    b = 1	# 初始化第二个斐波那契数b为1
    yield b	# 使用yield关键字返回第二个斐波那契数b
    while True:	# 当循环条件满足时,执行循环体
        a, b = b, a + b	# 更新斐波那契数列的下一个值
        yield b	# 使用yield关键字返回新的斐波那契数
        
f = fib() # 生成器	# 创建一个生成器对象f,用于调用fib函数
for i in range(6):	# 使用for循环遍历0到5的整数序列
    print(i+1, next(f))	# 打印当前整数i加1和对应的斐波那契数

# 返回结果:1 1
# 返回结果:2 1
# 返回结果:3 2
# 返回结果:4 3
# 返回结果:5 5
# 返回结果:6 8

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/136393.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

linux gdb 调试 常见调试命令介绍+总结

1.调试前准备 -g gcc arcg.c -g -oO -o app //必须添加-g 2.调试 gdb gdb app 3.常见调试命令 set args 1 2 3 4 5 6 //设置参数 show args //查看参数 3.1执行程序 1.start2. run gdb app set args 1 2 3 4 5 start //执行一行 c //继续执行 q…

【Git】Git分支与应用分支Git标签与应用标签

一&#xff0c;Git分支 1.1 理解Git分支 在 Git 中&#xff0c;分支是指一个独立的代码线&#xff0c;并且可以在这个分支上添加、修改和删除文件&#xff0c;同时作为另一个独立的代码线存在。一个仓库可以有多个分支&#xff0c;不同的分支可以独立开发不同的功能&#xff0…

【yolov5】onnx的INT8量化engine

GitHub上有大佬写好代码&#xff0c;理论上直接克隆仓库里下来使用 git clone https://github.com/Wulingtian/yolov5_tensorrt_int8_tools.git 然后在yolov5_tensorrt_int8_tools的convert_trt_quant.py 修改如下参数 BATCH_SIZE 模型量化一次输入多少张图片 BATCH 模型量化…

操作系统(二)内存管理的基础知识

文章目录 前言内存管理地址空间与地址生成连续内存分配内存碎片连续分配算法碎片整理 非连续内存分配虚拟内存管理虚拟内存地址内存分段内存分页段页式内存管理虚拟内存的覆盖技术虚拟内存的交换技术 缺页异常内存页面置换算法局部页面置换算法Belady现象全局页面置换算法抖动和…

Mybatis-Plus入门

Mybatis-Plus入门 MyBatis-Plus 官网&#xff1a;https://mp.baomidou.com/ 1、简介 MyBatis-Plus (简称 MP) 是一个 MyBatis 的增强工具&#xff0c;在 MyBatis 的基础上只做增强不做改变&#xff0c;为简化开发、 提高效率而生。 https://github.com/baomidou/mybatis-p…

【MySQL系列】第二章 · SQL(上)

写在前面 Hello大家好&#xff0c; 我是【麟-小白】&#xff0c;一位软件工程专业的学生&#xff0c;喜好计算机知识。希望大家能够一起学习进步呀&#xff01;本人是一名在读大学生&#xff0c;专业水平有限&#xff0c;如发现错误或不足之处&#xff0c;请多多指正&#xff0…

深入理解强化学习——多臂赌博机:知识总结

分类目录&#xff1a;《深入理解强化学习》总目录 我们在《深入理解强化学习——多臂赌博机》系列文章中介绍了几种平衡试探和开发的简单方法。 ϵ − \epsilon- ϵ−贪心方法在一小段时间内进行随机的动作选择&#xff0c;而UCB方法虽然采用确定的动作选择&#xff0c;却可以通…

Leetcode100128. 高访问员工

Every day a Leetcode 题目来源&#xff1a;100128. 高访问员工 解法1&#xff1a;模拟 把名字相同的员工对应的访问时间&#xff08;转成分钟数&#xff09;分到同一组中。 对于每一组的访问时间 accessTime&#xff0c;排序后&#xff0c;判断是否有 accessTime[i] - ac…

吃透 Spring 系列—Web部分

目录 ◆ Spring整合web环境 - Javaweb三大组件及环境特点 - Spring整合web环境的思路及实现 - Spring的web开发组件spring-web ◆ web层MVC框架思想与设计思路 ◆ Spring整合web环境 - Javaweb三大组件及环境特点 在Java语言范畴内&#xff0c;web层框架都是基于J…

win环境Jenkins部署前端项目

今天分享win环境Jenkins部署前端vue项目&#xff0c;使用的版本jenkins版本Jenkins 2.406版本。 前提是jenkins安装好了&#xff0c;通用配置已经配置好了&#xff0c;可以参考上两篇博客。 1、前端项目依赖nodejs&#xff0c;需要安装相关插件 点击进入 安装成功标准 jenki…

【Vue3】scoped 和样式穿透

我们使用很多 vue 的组件库&#xff08;element-plus、vant&#xff09;&#xff0c;在修改样式的时候需要进行其他操作才能成功更改样式&#xff0c;此时就用到了样式穿透。 而不能正常更改样式的原因就是 scoped 标记。 scoped 的渲染规则&#xff1a; <template>&l…

如何在ModelScope社区魔搭下载所需的模型

本篇文章介绍如何在ModelScope社区下载所需的模型。 若您需要在ModelScope平台上有感兴趣的模型并希望能下载至本地&#xff0c;则ModelScope提供了多种下载模型的方式。 使用Library下载模型 若该模型已集成至ModelScope的Library中&#xff0c;则您只需要几行代码即可加载…

STM32之DMA

一、DMA概述 DMA:直接寄存器访问 Direction:直接 Memory:存储器 Access:访问 就是一个外设用于搬运数据&#xff0c;就是一个搬运工。 在串口发送数据的时候&#xff1a;这种效率并不高 如何想要发送大量的数据的时候可以利用DMA 1、DMA工作流程 没有DMA参与…

【友提】2023年“思维100”编程比赛开始报名,名额有限报名抓紧

根据官方昨天发布的通知&#xff0c;2023年上海市“科学小公民”实践展示活动之“思维100”STEM应用能力编程活动&#xff08;秋季&#xff09;开始报名了&#xff0c;为便于大家了解&#xff0c;六分成长为大家整理关键信息如下。为便于叙述&#xff0c;该活动简称为思维100编…

RGB颜色空间与BMP格式图片

RGB颜色空间 RGB可以分为两大类&#xff1a;一种是索引形式&#xff0c;一种是像素形式&#xff1a; 索引形式&#xff1a;存储每个像素在调色板中的索引 RGB1&#xff1a;每个像素用1bit表示&#xff0c;调色板中只包含两种颜色&#xff08;黑白&#xff09;RGB4&#xff1a…

卸载本地开发环境,拥抱容器化开发

以前在公司的时候&#xff0c;使用同事准备的容器化环境&#xff0c;直接在 Docker 内进行开发&#xff0c;爽歪歪呀。也是在那时了解了容器化开发的知识&#xff0c;可惜了&#xff0c;现在用不到那种环境了。所以打算自己在本地也整一个个人的开发环境&#xff0c;不过因为我…

吴恩达《机器学习》8-3->8-4:模型表示I、模型表示II

8.3、模型表示I 一、大脑神经网络的基本原理 为了构建神经网络模型&#xff0c;首先需要理解大脑中的神经网络是如何运作的。每个神经元都可以被看作是一个处理单元或神经核&#xff0c;它包含多个输入&#xff08;树突&#xff09;和一个输出&#xff08;轴突&#xff09;。…

在vue3中使用Element-plus的图标

首先安装Element-Plus-icon # 选择一个你喜欢的包管理器# NPM $ npm install element-plus/icons-vue # Yarn $ yarn add element-plus/icons-vue # pnpm $ pnpm install element-plus/icons-vue 如何使用 Element-Plus-icon官方文档链接Icon 图标 | Element Plus (element-…

【操作系统】1.1 操作系统的基础概念、功能以及特性

&#x1f4e2;&#xff1a;如果你也对机器人、人工智能感兴趣&#xff0c;看来我们志同道合✨ &#x1f4e2;&#xff1a;不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 &#x1f4e2;&#xff1a;文章若有幸对你有帮助&#xff0c;可点赞 &#x1f44d;…

Unity之NetCode多人网络游戏联机对战教程(8)--玩家位置同步

文章目录 前言添加相机玩家添加对应组件服务端权威&#xff08;server authoritative&#xff09;客户端权威&#xff08;client authoritative&#xff09;服务端同步位置阅读与理解PlayerTransformSync.csNetworkVariableUploadTransformSyncTransform 后话 前言 承接上篇&a…