Pandas groupby 性能优化:实现高效数据聚合

DDD
发布: 2025-11-23 13:49:16
原创
713人浏览过

Pandas groupby 性能优化:实现高效数据聚合

本文深入探讨了pandas `groupby`操作在处理大规模数据时可能出现的性能瓶颈,特别是当结合`agg`方法进行多重聚合或使用自定义函数时。文章提出并详细演示了一种“懒惰式groupby”的优化策略,通过预先创建`groupby`对象,然后对每个列单独执行聚合操作,显著提升了数据聚合的效率。文中提供了具体的代码示例和性能对比,并讨论了如何灵活控制输出列名结构,旨在帮助用户编写更高效的pandas代码。

理解 Pandas groupby 的性能挑战

Pandas groupby 是数据分析中一个极其强大的工具,用于根据一个或多个键对数据进行分组,然后对每个组执行聚合操作。然而,当数据集规模增大,并且聚合操作变得复杂时,其性能可能会显著下降。常见的慢速模式之一是结合 agg 方法进行多列、多类型或包含自定义函数的聚合。

考虑以下一个典型的使用场景,我们希望根据 specimen 和 delta_t 列对数据进行分组,并计算 measuremnt 列的均值、75%分位数和最大值,以及 lag 列的均值。

import pandas as pd
import numpy as np

# 模拟数据集
data = {
    'delta_t': np.random.randint(0, 301, 100000), # 增加数据量以凸显性能问题
    'specimen': np.random.choice(['X', 'Y', 'Z'], 100000),
    'measuremnt': np.random.rand(100000),
    'lag': np.random.rand(100000)
}
df = pd.DataFrame(data)

# 定义一个自定义的75%分位数函数
def q75(x):
    return x.quantile(0.75)

# 原始的聚合代码
# df_result = df.groupby(['specimen', 'delta_t']).agg({
#     'measuremnt': ['mean', q75, 'max'],
#     'lag': 'mean'
# }).reset_index()
登录后复制

当上述代码应用于大型DataFrame时,用户可能会观察到执行时间随数据量呈非线性增长,导致处理效率低下。这通常是由于 agg 方法在内部处理多重聚合和自定义函数时,可能需要进行多次数据迭代或不必要的类型转换。

优化策略:懒惰式 groupby (Lazy Groupby)

为了解决 groupby().agg() 可能带来的性能问题,我们可以采用一种“懒惰式 groupby”的优化策略。其核心思想是:首先创建 groupby 对象,然后对该对象中的每个需要聚合的列单独执行聚合函数,最后将这些结果组合成一个新的DataFrame。这种方法通常能够显著减少内部开销,因为它允许Pandas更直接地优化每个独立的聚合操作。

1. 性能对比与优化实现

让我们通过具体的代码示例来展示这种优化方法及其带来的性能提升。

原始 agg 方法的性能(示例,假设数据量增大):

# %%timeit -n 10
# df_result_original = df.groupby(['specimen', 'delta_t']).agg({
#     'measuremnt': ['mean', q75, 'max'],
#     'lag': 'mean'
# }).reset_index()
# 假设其执行时间为 40-50 毫秒 (针对10万行数据)
登录后复制

优化后的“懒惰式 groupby”实现:

网龙b2b仿阿里巴巴电子商务平台
网龙b2b仿阿里巴巴电子商务平台

本系统经过多次升级改造,系统内核经过多次优化组合,已经具备相对比较方便快捷的个性化定制的特性,用户部署完毕以后,按照自己的运营要求,可实现快速定制会费管理,支持在线缴费和退费功能财富中心,管理会员的诚信度数据单客户多用户登录管理全部信息支持审批和排名不同的会员级别有不同的信息发布权限企业站单独生成,企业自主决定更新企业站信息留言、询价、报价统一管理,分系统查看分类信息参数化管理,支持多样分类信息,

网龙b2b仿阿里巴巴电子商务平台 0
查看详情 网龙b2b仿阿里巴巴电子商务平台
# %%timeit -n 10
groups = df.groupby(['specimen', 'delta_t'])

df_result_optimized = pd.DataFrame({
    'measurement_mean': groups['measuremnt'].mean(),
    'measurement_q75': groups['measuremnt'].quantile(.75),
    'measurement_max': groups['measuremnt'].max(),
    'lag_mean': groups['lag'].mean()
}).reset_index()

# 假设其执行时间为 2-3 毫秒 (针对10万行数据),性能提升显著
print(df_result_optimized.head())
登录后复制

输出示例:

  specimen  delta_t  measurement_mean  measurement_q75  measurement_max  lag_mean
0        X        0          0.490533         0.490533         0.490533  0.076840
1        X        1          0.472935         0.472935         0.472935  0.473552
2        X        2          0.449622         0.449622         0.449622  0.507664
3        X        3          0.469796         0.469796         0.469796  0.435753
4        X        4          0.509748         0.509748         0.509748  0.472064
登录后复制

从上述对比中可以看出,通过将聚合操作分解到各个列上执行,并直接构建新的DataFrame,我们能够实现数量级的性能提升。这种方法避免了 agg 在处理复杂聚合时可能产生的额外开销。

2. 处理多层索引输出 (MultiIndex)

原始的 agg 方法在聚合多列并使用多个聚合函数时,默认会生成一个多层列索引(MultiIndex)。如果你的应用场景需要这种结构,也可以通过“懒惰式 groupby”方法来实现。只需在构建DataFrame时,将列名定义为元组即可。

groups = df.groupby(['specimen', 'delta_t'])

df_result_multiindex = pd.DataFrame({
    ('measurement','mean'): groups['measuremnt'].mean(),
    ('measurement','q75'): groups['measuremnt'].quantile(.75),
    ('measurement','max'): groups['measuremnt'].max(),
    ('lag','mean'): groups['lag'].mean()
}).reset_index()

print(df_result_multiindex.head())
登录后复制

输出示例:

  specimen  delta_t measurement                      lag
                               mean       q75       max    mean
0        X        0     0.490533  0.490533  0.490533  0.076840
1        X        1     0.472935  0.472935  0.472935  0.473552
2        X        2     0.449622  0.449622  0.449622  0.507664
3        X        3     0.469796  0.469796  0.469796  0.435753
4        X        4     0.509748  0.509748  0.509748  0.472064
登录后复制

通过使用元组作为字典的键,Pandas在构建DataFrame时会自动识别并创建多层列索引,从而模拟 agg 的默认输出结构。

注意事项与最佳实践

  1. 选择合适的聚合方法:
    • 对于简单的单列或少量聚合,agg 方法通常足够方便且性能尚可。
    • 当聚合操作复杂(多列、多函数、自定义函数)且数据量较大时,优先考虑“懒惰式 groupby”策略。
    • 如果聚合函数是Pandas或NumPy的内置函数,它们通常是高度优化的。自定义Python函数(如本例中的 q75)可能会引入额外的性能开销,尤其是在 agg 内部。
  2. 避免不必要的 reset_index(): 如果不需要将分组键作为普通列,可以省略 reset_index(),直接使用 groupby 结果的索引作为DataFrame的索引,这可以节省一步操作。
  3. 数据类型优化: 确保DataFrame中的列使用最合适的数据类型。例如,整数列不应存储为浮点数,这可以减少内存占用并可能加速某些操作。
  4. 内存管理: 对于极大数据集,即使是优化后的 groupby 也可能消耗大量内存。考虑使用Dask等分布式计算库,或分块处理数据。
  5. 预计算: 如果某些中间结果可以预先计算或缓存,可以进一步提升整体性能。

总结

Pandas groupby 是数据处理的核心功能,但其性能并非一成不变。通过理解 agg 方法在复杂场景下可能带来的开销,并采纳“懒惰式 groupby”的优化策略,开发者可以显著提升大数据聚合的效率。这种方法不仅能够加速计算,还能提供更灵活的列名控制,从而更好地适应不同的数据分析需求。在实际项目中,根据具体的数据规模和聚合复杂度,选择最适合的 groupby 实现方式,是编写高效Pandas代码的关键。

以上就是Pandas groupby 性能优化:实现高效数据聚合的详细内容,更多请关注php中文网其它相关文章!

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号