Python的列表推导式差点搞垮我的服务器

一个凌晨三点的报警电话

事情发生在某个周六的凌晨三点。

我被一通电话吵醒。手机屏幕上显示的是公司的监控告警系统------CPU使用率飙升到98%,内存快爆了,服务器快要撑不住了。

我迷迷糊糊地打开电脑,看到日志里有一行代码正在疯狂运行:

ini 复制代码
result = [process_item(item) for item in huge_list]

一行列表推导式。

就这一行代码,把我8核16G的服务器拖到了崩溃的边缘。

你可能觉得我在夸张。但那天晚上,这行看似优雅、简洁的列表推导式,差点让我整个周末都泡汤。

今天我就把这个故事从头到尾讲一遍。从"为什么我觉得列表推导式很酷",到"为什么它差点搞垮我的服务器",再到"我怎么把它救回来的"。

列表推导式:我曾经的"心头好"

说实话,在出事之前,我是列表推导式的铁杆粉丝。

你看这种代码:

ini 复制代码
# 传统写法
squares = []
for i in range(10):
    squares.append(i ** 2)

# 列表推导式
squares = [i ** 2 for i in range(10)]

三行变一行,干净利落。谁不爱呢?

再复杂一点:

ini 复制代码
# 带条件的
even_squares = [i ** 2 for i in range(20) if i % 2 == 0]

# 两层循环
pairs = [(x, y) for x in range(5) for y in range(5)]

# 嵌套推导式
matrix = [[j for j in range(5)] for i in range(5)]

写起来顺手,读起来也直观。我当时觉得,这就是Python优雅的典范。

直到那个凌晨。

事故现场:到底发生了什么

让我还原一下当时的场景。

我的任务是从数据库里读取100万条用户记录,对每条记录做一些处理(格式化、校验、补充信息),然后生成一个报表。

数据量大概是这样:

  • 用户表:100万条记录
  • 每条记录处理后,会变成一个字典,包含大约50个字段
  • 最终结果是一个列表,里面装了100万个字典

我写的代码大致是这样的:

scss 复制代码
def process_user(user_data):
    # 模拟一些处理逻辑
    return {
        'id': user_data['id'],
        'name': user_data['name'].strip().title(),
        'email': user_data['email'].lower(),
        'score': calculate_score(user_data),
        'tags': parse_tags(user_data.get('tags', '')),
        # ... 还有40多个字段
    }

def get_report():
    users = db.fetch_all_users()  # 返回100万条记录
    result = [process_user(user) for user in users]
    return result

在测试环境,数据量只有1000条,这段代码跑得飞快,不到0.1秒就完成了。

但到了生产环境,100万条数据,情况完全不一样了。

问题出在哪里?两个地方。

问题一:内存爆炸

列表推导式会一次性把所有结果都放在内存里。

100万个字典,每个字典大约占用500字节(实际只多不少),那就是:

复制代码
1,000,000 × 500 ≈ 500,000,000 字节 ≈ 500 MB

这只是结果本身。别忘了原始数据 users 也还在内存里,还有中间过程中产生的各种临时对象。

实际内存占用,大概在1.5GB到2GB之间。

我的服务器只有16GB内存,看起来好像够用?但问题是,这个服务同时要处理多个请求。如果三个报表同时跑,内存直接炸。

问题二:CPU排队

列表推导式是单线程的。处理100万个用户,就是一个接一个地处理,处理完第一个才处理第二个。

每个用户处理需要多少时间?假设是0.5毫秒(其实业务逻辑往往更慢),那么总时间:

ini 复制代码
1,000,000 × 0.0005 = 500 秒 ≈ 8.3 分钟

一个报表要跑8分钟。用户早就等不及关页面了。

而在这8分钟里,CPU一直满负荷运转,其他请求都被堵在后面排队。

为什么列表推导式会这样

列表推导式本质上是一个语法糖。它做的事情,和你写一个 for 循环然后 append 是一样的。

ini 复制代码
# 这两种写法,内存和时间的消耗是一样的
result = [process(x) for x in data]           # 列表推导式

result = []                                    # 等价写法
for x in data:
    result.append(process(x))

两者都是:

  1. 创建一个空列表
  2. 遍历数据,每次处理一个元素
  3. 把结果追加到列表末尾
  4. 最后返回整个列表

所以当数据量大的时候,列表推导式的问题就很明显:

  • 内存:一次性存储所有结果
  • 速度:单线程串行处理

这不是列表推导式本身的问题,而是"一次性把所有数据装进列表"这个模式的问题。

那晚我是怎么救回来的

凌晨三点,我喝了杯凉水,开始改代码。

第一板斧:用生成器代替列表

生成器和列表推导式长得几乎一样,只是把方括号换成圆括号:

ini 复制代码
# 列表推导式:一次性生成所有结果
result_list = [process(x) for x in data]  # 占用大量内存

# 生成器表达式:按需生成结果
result_gen = (process(x) for x in data)   # 几乎不占内存

生成器不会一次性把所有结果算出来,而是"需要用的时候才算"。内存占用从几百MB降到了几乎可以忽略不计。

但是,生成器只能遍历一次。如果你要反复使用这些数据,生成器就不合适了。

对于我的报表场景,数据只输出一次,用生成器完美。

第二板斧:分批处理

有时候你必须得到一个完整的列表(比如需要反复使用、需要取长度、需要排序)。这时候怎么办?分批处理。

python 复制代码
def process_in_batches(data, batch_size=10000):
    """分批处理数据,避免一次性占用太多内存"""
    results = []
    for i in range(0, len(data), batch_size):
        batch = data[i:i+batch_size]
        batch_results = [process(item) for item in batch]
        results.extend(batch_results)
        
        # 可选:每批处理后打印进度
        print(f"已处理 {min(i+batch_size, len(data))}/{len(data)} 条")
    
    return results

这样做的好处:内存里最多同时存在 batch_size 个处理结果,而不是全部。

第三板斧:并行处理

处理速度慢的问题,需要用并行来解决。

Python的 concurrent.futures 模块提供了简单的并行方案:

python 复制代码
from concurrent.futures import ProcessPoolExecutor, as_completed

def process_in_parallel(data, process_func, max_workers=8):
    """多进程并行处理数据"""
    results = []
    
    with ProcessPoolExecutor(max_workers=max_workers) as executor:
        # 提交所有任务
        futures = {executor.submit(process_func, item): item for item in data}
        
        # 按完成顺序收集结果
        for future in as_completed(futures):
            try:
                result = future.result()
                results.append(result)
            except Exception as e:
                print(f"处理出错:{e}")
    
    return results

用8个进程并行处理,原本8分钟的工作,理论上可以缩短到1分钟左右。

但要注意:多进程有额外的开销(进程创建、数据序列化、结果反序列化)。如果每个任务的处理时间很短(比如几毫秒),那开多进程反而更慢。一般来说,单任务处理时间超过0.1秒,才值得用多进程。

更优雅的方案:看看你的数据流

经过那一晚,我重新思考了一个问题:我真的需要那个列表吗?

很多时候,我们需要的是一个可迭代的对象 ,而不是一个具体的列表

比如你要把数据写入文件:

css 复制代码
# 不要这样做
results = [process(x) for x in data]
for result in results:
    f.write(str(result) + '\n')

# 这样做
for x in data:
    f.write(str(process(x)) + '\n')

再比如你要把数据发送给API:

ini 复制代码
# 不要这样做
results = [process(x) for x in data]
api.send_batch(results)

# 这样做(如果API支持流式发送)
for x in data:
    api.send_one(process(x))

再比如你要计算统计值:

ini 复制代码
# 不要这样做
results = [process(x) for x in data]
average = sum(results) / len(results)

# 这样做(边计算边求和)
total = 0
count = 0
for x in data:
    total += process(x)
    count += 1
average = total / count

这些例子的共同点:你根本不需要同时保留所有结果

列表推导式什么时候该用,什么时候不该用

经过这次教训,我给自己定了几条规则。

该用列表推导式的场景:

  1. 数据量小(比如少于1万条),一眼能看出上限
  2. 结果列表确实需要反复使用
  3. 代码可读性的收益大于性能损耗
  4. 临时脚本、一次性数据处理

不该用列表推导式的场景:

  1. 数据量未知或可能很大(从数据库、文件、API读取)
  2. 内存受限的环境(如云函数、容器)
  3. 每个元素处理成本高(IO密集、计算密集)
  4. 结果只需要使用一次

可以改用生成器表达式的场景:

  1. 数据量大,但只需要遍历一次
  2. 需要链式处理多个转换步骤
  3. 不想在内存里囤积所有数据

生成器的写法:

ini 复制代码
# 列表推导式
result = [process(x) for x in data]

# 生成器表达式(语法几乎一样)
result = (process(x) for x in data)

一个快速判断的工具函数

有时候你写代码时不确定数据量有多大。这时候可以写一个智能版本:

python 复制代码
from collections.abc import Iterable

def smart_map(func, data, threshold=10000):
    """
    智能处理:小数据用列表推导式,大数据用生成器
    """
    if not isinstance(data, Iterable):
        raise TypeError("data must be iterable")
    
    # 如果数据有长度且小于阈值,返回列表
    if hasattr(data, '__len__') and len(data) < threshold:
        return [func(x) for x in data]
    
    # 否则返回生成器
    return (func(x) for x in data)

# 使用
result = smart_map(process_user, users)
# 如果 users 有长度且小于10000,result 是列表
# 否则 result 是生成器

这个函数帮你做自动判断,代码写起来不用纠结。

事故后的复盘

第二天上班,我做了几件事:

第一,给监控加了内存告警。之前只看CPU,内存问题完全被忽略了。

第二,给代码加了自动降级。当数据量超过阈值时,自动切换到流式处理模式。

第三,也是最重要的------重新理解了"优雅"的含义

我以前觉得代码越短越优雅。现在我觉得,能在正确的场景下正确运行的代码,才是真正的优雅

一行列表推导式看起来很酷。但如果它让你的服务器崩溃,那就不叫优雅,叫灾难。

最后的总结

列表推导式不是恶魔。它很好用,但你需要知道它的边界。

记住三句话:

  1. 列表推导式会一次性把所有结果塞进内存------数据量大时别用它
  2. 生成器表达式是它的替代品------把方括号换成圆括号就行
  3. 如果必须用列表,考虑分批处理或并行处理

那次事故之后,我再看到列表推导式,都会下意识问自己三个问题:

  • 这个数据有多大?
  • 我真的需要同时保留所有结果吗?
  • 换成生成器会不会更好?

这三个问题,也送给正在看文章的你。

你在代码里有没有被列表推导式坑过?或者有什么更奇葩的经历?欢迎在评论区聊聊。

相关推荐
学计算机的计算基1 小时前
2026 年 AI 助手三国杀:Claude Code vs 腾讯马维斯 vs MiniMax Mavis,我同时用了三周,结论很意外
java·人工智能·python·算法·langchain
我有2只猫1 小时前
LabelStudio二次开发
人工智能·python·django·ocr
石山代码1 小时前
Python 进阶学习指南
开发语言·python
用户8356290780512 小时前
Python 在 PowerPoint 中创建箱形图
后端·python
databook3 小时前
用SymPy自动求解三角形构造与全等条件验证
python·数学·动效
lunzi_fly3 小时前
【学习笔记】《Python编程 从入门到实践》第6章:字典创建、遍历与嵌套用法详解
python·python 小白学习
柒和远方4 小时前
LeetCode 452. 用最少数量的箭引爆气球 —— 区间贪心经典:排序 + 扫描一箭穿心
javascript·python·算法
winfredzhang5 小时前
用 Python + wxPython 做一个个人健康饮食管理工具:从记录三餐到综合生活建议
python·wxpython·deepseek·生活习惯管理
Irissgwe5 小时前
十、LangGraph能力详解:工作流的常见模式
python·langchain·ai编程·工作流·langgraph