当前位置: 首页
业界动态
处理 10GB 文件不爆内存,Python 迭代器凭什么

处理 10GB 文件不爆内存,Python 迭代器凭什么

热心网友 时间:2026-04-16
转载

1. 场景案例:Python大文件处理

去年遇到一个数据清洗任务,日志文件足足有12GB。同事的第一反应很直接:用readlines()把文件全部读进内存,再逐条处理。结果程序跑了三分钟,直接报了个MemoryError

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

其实只改了一个地方:把readlines()换成直接遍历文件对象。就这么一个调整,内存峰值从接近14GB降到了不足100MB。

背后的区别很简单:前者是一次性加载,后者则利用了Python迭代器的惰性特性。很多开发者写了多年Python,依然习惯用列表推导式处理一切。不是说不能用,但遇到大数据量时,迭代器和生成器才是那个“正解”。

2. 什么是迭代器

用大白话说,迭代器就是一个知道怎么“一个一个给”东西的对象。你每次问它要下一个,它就给你下一个;给完了,就抛个信号说“没了”,整个过程干净利落。

Python里那个熟悉的for x in something:语法,底层调用的正是迭代器协议。这个协议的核心,就两个方法:

  • __iter__:返回迭代器对象本身。
  • __next__:返回下一个元素。如果没元素了,就抛出StopIteration异常。

看,这就是迭代器协议的全部内容,两个方法,简单直接。

class Counter:
    def __init__(self, max_num):
        self.max_num = max_num
        self.current = 0

    def __iter__(self):
        return self

    def __next__(self):
        if self.current >= self.max_num:
            raise StopIteration
        self.current += 1
        return self.current

for i in Counter(5):
    print(i)  # 输出:1, 2, 3, 4, 5

上面这个Counter类就是一个自定义迭代器。__iter__返回自己,__next__每次递增并返回当前值,到上限就抛出异常结束迭代。

3. 可迭代对象 vs 迭代器

这两个概念经常被混用,但它们确实不是一回事。

  • 可迭代对象:拥有__iter__()方法,能返回一个迭代器。比如列表(list)、元组(tuple)、字符串(str)、字典(dict)。
  • 迭代器:不仅拥有__iter__()方法,还有__next__()方法,本身就能被遍历。

举个例子,列表是可迭代对象,但它本身不是迭代器。你可以用for循环遍历它,但不能直接对它调用next()

lst = [1, 2, 3]
next(lst)  # TypeError: list object is not an iterator

it = iter(lst)  # 通过iter()函数获取它的迭代器
next(it)  # 1
next(it)  # 2
next(it)  # 3
next(it)  # 抛出StopIteration异常

这里的iter()函数,其实就是调用了对象内部的__iter__()方法。

简单总结一下:

  • 可迭代对象不一定是迭代器。
  • 迭代器一定是可迭代对象。
  • 可迭代对象可以通过iter()函数得到一个迭代器。

4. 生成器:迭代器的快捷方式

实现一个完整的迭代器类需要定义两个方法,略显繁琐。而生成器提供了一种更简洁的实现方式,它会自动满足迭代器协议。

创建生成器,最常用的方式就是使用yield关键字。一个函数里只要包含了yield,它就变成了生成器函数。

def counter(max_num):
    current = 0
    while current < max_num:
        current += 1
        yield current

for i in counter(5):
    print(i)  # 输出:1, 2, 3, 4, 5

yieldreturn的关键区别在于:return执行完函数就彻底结束了,而yield会暂停函数执行,保存当前状态,下次调用next()时再从暂停的地方继续。这正是生成器节省内存的秘诀——它不事先存储所有结果,而是“用一次,算一次”。

(1) 生成器表达式

还有一种更轻量的创建方式:生成器表达式。只需把列表推导式的方括号[]换成圆括号()

# 列表推导式 - 一次性生成所有结果,全部加载到内存
squares = [x*x for x in range(1000000)]  # 占内存约8.06MB

# 生成器表达式 - 惰性计算,用一次算一次
squares = (x*x for x in range(1000000))  # 内存占用极小,约200字节

当数据量达到一定规模时,这两种方式的区别就不再是“快一点”或“慢一点”,而是“能跑”和“根本跑不了”的天壤之别。

5. 实战场景

(1) 场景一:大文件处理

# 错误写法:一次性加载
with open('huge.log') as f:
    lines = f.readlines()  # 返回list,全部加载到内存
    for line in lines:
        process(line)

# 正确写法:利用文件对象自身的迭代器特性
with open('huge.log') as f:
    for line in f:  # 文件对象是迭代器,每次只读一行
        process(line)

Python的文件对象本身就是一个迭代器,直接遍历它,解释器会自动按行惰性读取,避免内存爆炸。

(2) 场景二:数据流处理

def read_api_pages(url):
    page = 1
    while True:
        response = requests.get(url, params={'page': page})
        data = response.json()
        if not data:
            break
        yield from data  # 将子迭代器的值直接产出
        page += 1

for item in read_api_pages('https://api.example.com/items'):
    process(item)

这里用到了yield from,它能将子迭代器的值直接传递出来。这种模式特别适合处理分页API、数据库游标等需要连续获取数据的场景。

(3) 场景三:管道式数据处理

def read_csv(filepath):
    with open(filepath) as f:
        for line in f:
            yield line.strip().split(',')

def filter_age(records, min_age):
    for record in records:
        if int(record[2]) >= min_age:
            yield record

def format_output(records):
    for record in records:
        yield f"{record[0]}|{record[1]}|{record[2]}"

# 像组装流水线一样组合起来
data = read_csv('users.csv')
filtered = filter_age(data, 18)
formatted = format_output(filtered)

for line in formatted:
    print(line)

每个函数都是一个生成器,数据像在流水线(pipeline)中传递。中间不需要任何额外的列表或容器来存储临时结果,内存效率极高。

6. 常见坑

第一个坑:迭代器和生成器是“一次性消耗品”。

gen = (x for x in range(5))
list(gen)  # 第一次转换,得到 [0, 1, 2, 3, 4]
list(gen)  # 第二次转换,得到 [] —— 因为生成器已经耗尽了

遍历完一次后,迭代器或生成器就空了。如果需要重复使用,要么将其转换为列表,要么重新创建。

第二个坑:生成器函数返回的是生成器对象,不是执行结果。

def my_gen():
    yield 1
    yield 2
    yield 3

result = my_gen()  # 这里只是得到了一个生成器对象,函数体内的代码并未执行
# 需要遍历(如用for循环或next())才会触发yield语句执行

新手常会困惑:明明调用了函数,为什么没有输出?记住,调用生成器函数返回的是一个待“激活”的生成器对象,真正的计算发生在迭代时。

7. 总结

最后,我们来梳理一下这几个核心概念的关系:

  • 可迭代对象 (Iterable):有__iter__()方法。
  • 迭代器 (Iterator):继承自可迭代对象,多了__next__()方法。__iter__()通常返回self,并能保持迭代状态。文件对象就在这里。
  • 生成器 (Generator):一种特殊的迭代器,由生成器函数或生成器表达式创建。除了迭代器的方法,还有send(), throw(), close()等方法,使用yield关键字实现。

归根结底,迭代器和生成器的核心优势在于惰性:惰性访问、惰性计算,用一次算一次,对内存极其友好。它们的协议也足够简单,实现__iter____next__两个方法就能搞定。生成器则是更便捷的语法糖,yield让一个普通函数变身迭代器。

处理小数据时,列表和生成器的差异或许微不足道。可一旦数据量上来,这其中的区别,往往就决定了程序是顺利运行,还是中途崩溃。

文章开头那个12GB的日志文件案例,后来成了技术面试中的一个经典问题。题目本身不难,但能清晰地分辨出一个开发者,是否真正理解了Python迭代器与生成器的原理,以及能否在关键时刻做出正确的应用选择。

来源:https://www.51cto.com/article/840720.html

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
日产重塑英菲尼迪:弃独立平台,推7款新车强化差异化

日产重塑英菲尼迪:弃独立平台,推7款新车强化差异化

日产重塑英菲尼迪战略:放弃独立平台,聚焦7款新车打造核心差异化 4月16日,日产汽车首席执行官内田诚(Makoto Uchida)对英菲尼迪品牌的现状与发展路径进行了深入剖析。他指出,这一豪华品牌近年来面临的市场挑战,主要源于过往关键战略的调整,目前品牌正处在全面重塑与战略转型的关键时期。 内田诚直

时间:2026-04-16 13:10
沃尔沃迎99周年:发布新型头颈保护座椅,女性颈椎伤风险降50%

沃尔沃迎99周年:发布新型头颈保护座椅,女性颈椎伤风险降50%

沃尔沃迎99周年:发布新型头颈保护座椅,女性颈椎伤风险降50% 2026年4月16日,沃尔沃汽车迎来了品牌创立99周年的重要里程碑。选择在这个极具纪念意义的时刻发布全新安全技术,深刻诠释了其将“安全”这一核心品牌基因不断推向新境界的决心。发布会上,沃尔沃汽车安全防护领域高级技术专家洛塔·雅各布森女士

时间:2026-04-16 13:07
Pinia 全新版本!以后不再需要 Axios!

Pinia 全新版本!以后不再需要 Axios!

Pinia Colada:为Vue应用注入优雅的数据请求体验 在Vue js应用开发过程中,高效管理数据请求是提升开发效率和保障代码质量的核心环节。Pinia Colada正是为此而设计的解决方案。作为Pinia生态体系中的上层封装库,其核心目标非常明确:彻底消除数据请求环节中重复的模板代码与复杂逻

时间:2026-04-16 13:02
沃尔沃全新EX90纯电旗舰SUV发布:2+3+2式七座布局、双电机四驱,预售价53.99万元起

沃尔沃全新EX90纯电旗舰SUV发布:2+3+2式七座布局、双电机四驱,预售价53.99万元起

沃尔沃全新EX90纯电旗舰SUV发布:2+3+2式七座布局、双电机四驱,预售价53 99万元起 值此沃尔沃品牌创立99周年之际,其备受期待的全新纯电旗舰SUV——EX90于今日正式开启预售。新车基于先进的SPA2原生豪华纯电平台打造,首发推出双电机四驱版Plus与双电机四驱高性能版Ultra两款车型

时间:2026-04-16 12:56
哪家 GEO 优化服务商专业?2026 年 4 月推荐评测口碑对比 TOP7 服务排名领先电商品牌流量转化困境

哪家 GEO 优化服务商专业?2026 年 4 月推荐评测口碑对比 TOP7 服务排名领先电商品牌流量转化困境

生成式AI时代,品牌如何抢占“语义主权”?聚焦三家值得关注的GEO服务商 生成式AI的浪潮正以前所未有的速度重塑企业与用户的连接界面。如今,大型语言模型日益成为人们获取信息、辅助决策的首选入口。在此背景下,GEO(生成式引擎优化)的重要性不言而喻——它不再只是一种技术优化,而是品牌在AI时代争夺“语

时间:2026-04-16 12:53
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜
热门教程
更多
  • 游戏攻略
  • 安卓教程
  • 苹果教程
  • 电脑教程