百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

python 入门到脱坑 python中的“器”—生成器

wptr33 2025-07-03 01:13 26 浏览

Python生成器(Generator)入门详解

生成器是Python中一种特殊的迭代器,它可以让你按需生成值而不是一次性计算所有值,非常适合处理大数据流或无限序列。

一、生成器基础概念

1. 生成器是什么?

  • 惰性计算:只在需要时生成值,节省内存
  • 状态保持:记住上次执行的位置
  • 可迭代:可以用在for循环中

2. 生成器 vs 普通函数

特性

普通函数

生成器函数

返回值

return一次

yield多次

执行

一次性执行完

可暂停/恢复

内存

存储所有结果

只存储当前状态

二、创建生成器的两种方式

1. 生成器函数(使用yield)

def count_up_to(max):
    count = 1
    while count <= max:
        yield count  # 暂停并返回值
        count += 1

# 使用生成器
counter = count_up_to(5)
for num in counter:
    print(num)  # 输出1, 2, 3, 4, 5

2. 生成器表达式(类似列表推导式)

# 列表推导式(立即计算)
squares_list = [x**2 for x in range(5)]  # [0,1,4,9,16]

# 生成器表达式(惰性计算)
squares_gen = (x**2 for x in range(5))  
print(list(squares_gen))  # [0,1,4,9,16](需要时计算)

三、生成器核心特性

1. 状态保持

def simple_generator():
    yield "第一次暂停"
    yield "第二次暂停"
    yield "最后一次"

gen = simple_generator()
print(next(gen))  # "第一次暂停"
print(next(gen))  # "第二次暂停"
print(next(gen))  # "最后一次"
# print(next(gen))  # 触发StopIteration

2. 只能遍历一次

numbers = (x for x in range(3))
print(list(numbers))  # [0,1,2]
print(list(numbers))  # [](已耗尽)

3. send()方法传递值

def generator_with_send():
    print("启动")
    x = yield "请发送值"
    print(f"收到: {x}")
    yield "结束"

g = generator_with_send()
print(next(g))       # 输出"启动"和"请发送值"
print(g.send(42))    # 发送42,输出"收到:42"和"结束"

四、实际应用场景

1. 处理大型文件

def read_large_file(filename):
    with open(filename, 'r') as f:
        for line in f:
            yield line.strip()  # 逐行生成

# 使用示例
for line in read_large_file('huge_data.txt'):
    process(line)  # 内存友好

2. 生成无限序列

def fibonacci():
    a, b = 0, 1
    while True:
        yield a
        a, b = b, a + b

# 获取前10个斐波那契数
fib = fibonacci()
first_10 = [next(fib) for _ in range(10)]

3. 数据管道处理

def integers():
    i = 1
    while True:
        yield i
        i += 1

def squares(seq):
    for num in seq:
        yield num * num

# 组合生成器
pipe = squares(integers())
for i in range(5):
    print(next(pipe))  # 1,4,9,16,25

五、生成器高级技巧

1. yield from(委托子生成器)

def sub_generator():
    yield 1
    yield 2

def main_generator():
    yield "开始"
    yield from sub_generator()  # 委托给子生成器
    yield "结束"

print(list(main_generator()))  # ['开始',1,2,'结束']

2. 生成器与异常处理

def generator_with_try():
    try:
        yield "正常执行"
    except ValueError as e:
        yield f"捕获到异常: {e}"
    finally:
        yield "总是执行"

g = generator_with_try()
print(next(g))  # "正常执行"
print(g.throw(ValueError("测试")))  # "捕获到异常: 测试"
print(next(g))  # "总是执行"

六、生成器常见问题

Q1: 生成器和迭代器有什么区别?

  • 生成器是一种特殊迭代器,使用更简洁的语法(yield)
  • 所有生成器都是迭代器,但迭代器不一定是生成器

Q2: 什么时候应该使用生成器?

  • 处理大数据集(避免内存不足)
  • 需要生成无限序列
  • 构建数据处理管道

Q3: 如何重启一个已耗尽的生成器?

def my_gen():
    yield 1
    yield 2

# 必须重新创建生成器
g1 = my_gen()
list(g1)  # [1,2]
g2 = my_gen()  # 新的生成器
list(g2)  # [1,2]

七、性能对比

操作

列表

生成器

创建1百万数字

存储所有元素

只存储算法

内存占用

~8MB

~1KB

访问速度

快速随机访问

顺序访问

建议:数据量大时优先使用生成器

相关推荐

oracle数据导入导出_oracle数据导入导出工具

关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...

继续学习Python中的while true/break语句

上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个else解...

python continue和break的区别_python中break语句和continue语句的区别

python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...

简单学Python——关键字6——break和continue

Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...

2-1,0基础学Python之 break退出循环、 continue继续循环 多重循

用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...

Python 中 break 和 continue 傻傻分不清

大家好啊,我是大田。今天分享一下break和continue在代码中的执行效果是什么,进一步区分出二者的区别。一、continue例1:当小明3岁时不打印年龄,其余年龄正常循环打印。可以看...

python中的流程控制语句:continue、break 和 return使用方法

Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...

L017:continue和break - 教程文案

continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...

作为前端开发者,你都经历过怎样的面试?

已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...

面试被问 const 是否不可变?这样回答才显功底

作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...

2023金九银十必看前端面试题!2w字精品!

导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。答案:CSS的盒模型是用于布局和定位元素的概念。它由内容区域...

前端面试总结_前端面试题整理

记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...

由浅入深,66条JavaScript面试知识点(七)

作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录由浅入深,66条JavaScript面试知识点(一)由浅入深,66...

2024前端面试真题之—VUE篇_前端面试题vue2020及答案

添加图片注释,不超过140字(可选)1.vue的生命周期有哪些及每个生命周期做了什么?beforeCreate是newVue()之后触发的第一个钩子,在当前阶段data、methods、com...

今年最常见的前端面试题,你会做几道?

在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...