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

Python生成器入门:用“按需生产”思维处理海量数据

off999 2025-05-23 19:15 22 浏览 0 评论

想象你要制作1000个蛋糕:

  • 列表的做法:先把1000个蛋糕全做好堆在厨房里,占满空间还容易变质
  • 生成器的做法:有人下单时才做一个蛋糕,厨房永远只有1个蛋糕,不占空间还新鲜

这就是Python生成器的核心逻辑——按需生成数据,绝不提前囤积。本文将用漫画式思维和极简代码,带新手彻底理解这个“内存救星”。

一、为什么列表处理大数据会“卡爆”?

案例:生成1亿个数字

# 用列表存储1亿个数字(灾难现场!)
big_list = [i for i in range(100000000)]  # 电脑可能直接卡死

问题根源

  • 列表像一个“超级仓库”,必须把所有数据一次性存进去
  • 数据量越大,仓库越占空间,小电脑根本扛不住!

二、生成器:数据界的“自动贩卖机”

生成器就像一台自动贩卖机:

  • 你按下按钮(调用next())
  • 贩卖机现做一份零食(生成一个数据)
  • 绝不提前生产堆在机器里

1. 生成器表达式:贩卖机的简易版

语法

(数据生成规则 for 原材料 in 供应列表)  # 用()代替列表的[]

示例:生成1-5的平方贩卖机

square_gen = (x**2 for x in range(1, 6))  # 先造一台贩卖机,不生产任何数据

如何买零食?

print(next(square_gen))  # 按下按钮:得到1(贩卖机生产第1个数据)
print(next(square_gen))  # 再按按钮:得到4(生产第2个数据)
# 每次调用next(),贩卖机才生产一个数据,内存始终只有1个数据!

2. 生成器函数:贩卖机的定制版

如果需要复杂的生产流程(如筛选、计算),可以用函数造贩卖机:
语法

def 贩卖机函数():
    for 原材料 in 供应列表:
        if 原材料符合条件:
            yield 加工后的产品  # yield是“出货口”,每次出一个产品后暂停

示例:生成100以内的偶数贩卖机

def even_gen():
    for x in range(100):
        if x % 2 == 0:
            yield x  # 出货口:给我一个偶数!

gen = even_gen()  # 造好贩卖机
print(next(gen))  # 0(第1次出货)
print(next(gen))  # 2(第2次出货)

三、生成器的超能力:内存占用永不爆炸

实验对比:列表vs生成器的内存消耗

数据量

列表占用内存(Python实测)

生成器占用内存

10个数字

约0.04KB

约0.0008KB

100万个数字

约40KB

约0.0008KB

1亿个数字

约4000KB(4MB)

约0.0008KB

原因

  • 列表:存1亿个数字 → 每个数字占4字节 → 总内存4亿字节=40MB
  • 生成器:只存“从0开始,每次+1”的规则 → 仅需几十字节记录规则

四、生成器的3个经典使用场景

场景1:处理超大文件(如10GB日志)

传统方法(错误示范)

with open('big.log', 'r') as f:
    lines = f.readlines()  # 把10GB文件全读进内存,电脑直接罢工!

生成器方法(正确做法)

def read_large_file(file_path):
    with open(file_path, 'r') as f:
        for line in f:  # 文件对象本身就是生成器,逐行读取
            yield line.strip()  # 每次只存1行数据,内存稳如老狗!

# 逐行处理:统计包含"ERROR"的行数
error_count = 0
for line in read_large_file('big.log'):
    if "ERROR" in line:
        error_count += 1

场景2:实时生成数据(如模拟传感器)

import time

def live_sensor():
    while True:
        yield time.strftime("%H:%M:%S")  # 无限生成当前时间
        time.sleep(1)  # 每秒生成1个数据

# 使用:打印实时时间,不存历史数据
sensor = live_sensor()
print(next(sensor))  # 15:30:01
print(next(sensor))  # 15:30:02
# 内存始终只存最新1个时间,绝不积压!

场景3:数据流水线(过滤→转换→分析)

# 生成器链:数据像流水一样逐个处理
def generate_data():
    yield 1; yield 2; yield 3; yield 4  # 生成原始数据

def filter_even(data):
    for x in data:
        if x % 2 == 0:
            yield x  # 过滤偶数

def convert_to_str(data):
    for x in data:
        yield f"数字{x}"  # 转换格式

# 流水线:生成→过滤→转换
pipeline = convert_to_str(filter_even(generate_data()))
for item in pipeline:
    print(item)  # 输出:"数字2", "数字4"

五、新手必看!生成器的3个“不能做”

不能做1:用索引访问生成器

gen = (x for x in range(5))
print(gen[2])  #  报错!生成器不能像列表一样用索引
# 正确做法:用for循环遍历或next()逐个获取

不能做2:重复使用耗尽的生成器

gen = (x for x in range(3))
print(next(gen))  # 0
print(next(gen))  # 1
print(next(gen))  # 2
print(next(gen))  #  报错!生成器已空
# 解决办法:重新创建生成器对象 gen = (x for x in range(3))

不能做3:用生成器存储需要反复访问的数据

#  错误场景:需要多次遍历数据
gen = (x for x in range(5))
print(sum(gen))  # 10(第一次遍历,生成器耗尽)
print(sum(gen))  # 0(第二次遍历,没有数据了!)

# 正确做法:用列表存储需要反复使用的数据
lst = [x for x in range(5)]
print(sum(lst))  # 10
print(sum(lst))  # 10(列表数据一直都在)

六、实战练习:用生成器计算10000以内的质数

需求:生成10000以内的所有质数,要求内存占用小于1MB
步骤1:定义质数判断函数

def is_prime(n):
    if n < 2:
        return False
    for i in range(2, int(n**0.5)+1):
        if n % i == 0:
            return False
    return True

步骤2:用生成器函数生成质数

def prime_generator(max_num):
    for n in range(2, max_num+1):
        if is_prime(n):
            yield n  # 是质数就出货!

步骤3:遍历生成器并统计

primes = prime_generator(10000)
print(f"10000以内的质数有:{list(primes)}")
# 内存占用:生成器全程仅存储当前判断的数字,远小于1MB!

总结:生成器的“三句口诀”

  1. 列表是仓库,生成器是流水线:列表存所有数据,生成器逐个造数据
  2. 用for循环替代next():for x in 生成器: 比手动调用next()更简单
  3. 大数据用生成器,小数据用列表:处理几万个数据用列表,百万级数据用生成器

给新手的练习建议

  1. 用生成器表达式生成1-100的奇数序列
  2. 尝试用生成器函数实现斐波那契数列(提示:用a, b = b, a+b)
  3. 思考:如何用生成器处理CSV文件的逐行解析?

生成器是Python中“聪明处理数据”的关键工具,学会它能让你在处理大数据时游刃有余。下次我们将深入迭代器原理,揭开生成器背后的神秘面纱,记得关注哦!

相关推荐

兄弟打印机怎么连接wifi(兄弟打印机怎么连接wifi手机打印)
  • 兄弟打印机怎么连接wifi(兄弟打印机怎么连接wifi手机打印)
  • 兄弟打印机怎么连接wifi(兄弟打印机怎么连接wifi手机打印)
  • 兄弟打印机怎么连接wifi(兄弟打印机怎么连接wifi手机打印)
  • 兄弟打印机怎么连接wifi(兄弟打印机怎么连接wifi手机打印)
uefi模式下找不到硬盘(uefi引导找不到硬盘)

首先你的安装盘必须是从UEFI启动的,然后它才能安装为UEFI启动。(条件:Fat32文件系统,efi文件夹)其次你MBR+BIOS的系统想换成GPT+EFI的,分区得做一点改动,腾出来100M的空...

win7怎么安装蓝牙驱动程序(win7电脑安装蓝牙驱动教程)

方法如下:  1、再开始里点击控制版面,点击【硬件和声音】找到【添加设备】  2、之后再选择你要添加的蓝牙耳机。  3、系统就会提示正在与蓝牙适配器连接,然后提示添加成功。  4、点击“开始”-“...

怎么装系统win7旗舰版(电脑怎么装win7旗舰版)

1、目前支持64位的Wincc版本有:WinccV7Sp3、WinccV11Sp2、WinccV12。2、Wincc的V11与V12两个版本不能共存,即不能同时安装在同一台电脑上。上述这两...

u盘恢复工具免费版(u盘文件恢复工具免费版)
  • u盘恢复工具免费版(u盘文件恢复工具免费版)
  • u盘恢复工具免费版(u盘文件恢复工具免费版)
  • u盘恢复工具免费版(u盘文件恢复工具免费版)
  • u盘恢复工具免费版(u盘文件恢复工具免费版)
qq特别关心查询入口电脑版(电脑版qq特别关心网站登录)
  • qq特别关心查询入口电脑版(电脑版qq特别关心网站登录)
  • qq特别关心查询入口电脑版(电脑版qq特别关心网站登录)
  • qq特别关心查询入口电脑版(电脑版qq特别关心网站登录)
  • qq特别关心查询入口电脑版(电脑版qq特别关心网站登录)
怎样恢复桌面图标原样(怎样恢复桌面图标原样显示)

桌面图标恢复原样步骤方法如下:1.右键点击图标、并点击‘属性’两字,再点击‘打开文件位置‘2.然后找到文件所在的位置,然后暂时将文件在一旁放着3.然后再去点击‘更换图标’这四个字,找到刚刚打开的文件...

消除u盘写保护(怎么解除u盘写保护)

如果您的U盘启用了写保护功能,可以尝试以下几种方法来解除写保护:1.检查U盘上的物理写保护开关:部分U盘有一个物理的写保护开关,可以在U盘的外壳上找到。请确保开关处于非写保护状态。2.使用CMD命...

microsoft国际版(微软国际版bing)

win10系统自带的edge浏览器不分国内版和国际版。搜索引擎Bing分为国内版和国际版,区别如下一、迎合不同人群的需求。国际版显得更加个性,他迎合了不同人群的需求,在收缩上更加方便化和智能化,后期可...

为什么格式化sd卡后不能用了

SD卡受损后不格式化直接修复的方法:1,当系统发现SD卡受损之后,会提示进行格式化。2,把SD卡插入读卡器,接到电脑USB后,电脑提示格式化,点取消。然后查看一下属性。直接使用属性中的工具-开始检查...

c盘删除的东西能恢复吗(c盘删掉的东西还在吗)

    1.重新安装。  如果是安装在c盘下的普通应用程序,重新下载安装即可;  2.备份恢复或重装系统。  如果是删除了系统文件,在系统备过份且知道删除...

电脑如何重装系统winxp(电脑如何重装xp系统教程)

重装XP系统的步骤如下:1.首先备份您的重要文件和数据。2.插入XP系统安装光盘,重新启动计算机。3.在启动时按下计算机制造商指定的键(通常是F2、F12、Esc等)进入BIOS设置或启动菜单。...

win8密钥永久激活工具(win8密钥永久激活工具在哪)

win8.1正式版升级win10,自动激活,win8.1的密钥不能激活win10,升级win10正式版以后,无论怎么安装win10,系统都是自动激活状态。终极PID检测产品密钥:GCRJD...

制作系统安装u盘教程(制作系统安装盘及系统安装的步骤)

答u盘做系统详细步骤如下:  1.首先我们要准备一个10G内存的空白u盘,然后在装机吧官网下载一个装机吧一键重装系统软件,安装完成后打开软件,制作模式选择”USB-HDD“分区格式为”NTFS“点击...

联想win7怎么进入bios设置(联想win7进入bios设置win10)
联想win7怎么进入bios设置(联想win7进入bios设置win10)

联想电脑win7进入bios设置的具体步骤如下: 1、首先我们打开电脑的同时,按下键盘上的“F2”。2、然后我们在弹出来的窗口中就可以进入到BIOS界面中。3、然后我们按下键盘上的“F10”,之后回车确定即可退出。联想电脑win7...

2025-11-09 14:03 off999

取消回复欢迎 发表评论: