0

0

利用Pandas高效计算DataFrame中指定日期前后数据总和

心靈之曲

心靈之曲

发布时间:2025-08-14 19:50:02

|

342人浏览过

|

来源于php中文网

原创

利用Pandas高效计算DataFrame中指定日期前后数据总和

本教程详细介绍了如何使用Pandas库高效处理DataFrame数据,根据每行中指定的日期,将日期列的数据分为“之前”和“之后”两部分,并分别计算这两部分的数值总和。通过数据重塑(melt)、条件判断(np.where)和分组聚合(groupby.sum)等操作,实现对复杂时间序列数据的灵活统计分析,适用于需要按特定时间点进行数据汇总的场景。

在处理包含时间序列数据的dataframe时,我们经常会遇到需要根据某个特定日期对数据进行分组汇总的需求。例如,在一个包含多个月份销售额的表格中,每行代表一个产品,并指定一个“参考日期”,我们需要计算该产品在参考日期“之前”和“之后”的销售总额。本教程将详细阐述如何利用pandas库的强大功能,优雅地解决这类问题。

场景描述

假设我们有一个DataFrame,其结构如下所示:

  • Code 列:代表唯一的标识符(如产品代码)。
  • Date 列:每行一个特定的参考日期,用于划分时间段。
  • 其他列:以日期命名,存储对应日期下的数值(如销售额、库存量等)。

我们的目标是为每行数据,基于其Date列的值,计算所有日期列中早于Date的数值总和(Before)以及晚于或等于Date的数值总和(After),并将这两个总和作为新列添加到原始DataFrame中。

核心思路

解决此问题的关键在于数据重塑(Reshaping)、条件判断和分组聚合。具体步骤如下:

  1. 数据重塑(Melt):将宽格式的日期列转换为长格式,使得所有日期值列的数据都集中到一个“值”列中,同时保留原始的Code和Date列以及新的“变量”列(原日期列名)。
  2. 数据类型统一:确保所有日期相关的列(包括Date和重塑后的“变量”列)的数据类型一致,以便进行比较。通常转换为字符串或日期时间类型。
  3. 条件判断与分类:根据每行的Date值与重塑后的“变量”列(即原始的日期列名)进行比较,判断该变量属于“之前”还是“之后”的分类。
  4. 分组聚合:按Code和新生成的“之前/之后”分类进行分组,并对“值”列进行求和。
  5. 结果重塑(Unstack):将“之前/之后”分类从行索引转换为列,形成Before和After两列。
  6. 合并结果:将计算出的Before和After列合并回原始DataFrame。

步骤详解与代码实现

首先,我们创建一个示例DataFrame来模拟上述场景:

Noya
Noya

让线框图变成高保真设计。

下载
import pandas as pd
import numpy as np

# 创建示例DataFrame
data = {
    'Code': ['12345', '12346', '12347'],
    '202001': [1000, 999, 1983],
    '202002': [1001, 1000, 1984],
    '202003': [1002, 1001, 1985],
    '202004': [1003, 1002, 1986],
    '202005': [1004, 1003, 1987],
    '202006': [1005, 1003, 1988],
    '202007': [3006, 1005, 1989],
    '202008': [1007, 1006, 1990],
    '202009': [1008, 1007, 1991],
    '202010': [1009, 1008, 1992],
    '202011': [1010, 1009, 1993],
    '202012': [1011, 1010, 1994],
    'Date': ['202004', '202006', '202010']
}
df = pd.DataFrame(data)

print("原始DataFrame:")
print(df)

核心处理代码:

# 1. 数据重塑 (melt)
# 将除 'Code' 和 'Date' 以外的所有列转换为长格式
tmp = df.melt(['Code', 'Date'])

# 2. 数据类型统一
# 确保 'Date' 列和 'variable' (原始日期列名) 列都是字符串类型,便于比较
tmp = tmp.astype({'Date': str, 'variable': str})

# 3. 条件判断与分类 (assign with np.where)
# 比较 'Date' (每行的参考日期) 和 'variable' (当前数据点的日期)
# 如果 'Date' 大于 'variable',则为 'Before',否则为 'After'
tmp = tmp.assign(col=lambda d: np.where(d['Date'].gt(d['variable']), 'Before', 'After'))

# 4. 分组聚合 (groupby.sum)
# 按 'Code' 和新生成的 'col' (Before/After) 分组,并对 'value' 求和
tmp = tmp.groupby(['Code', 'col'])['value'].sum()

# 5. 结果重塑 (unstack)
# 将 'col' 从行索引转换为列,形成 'Before' 和 'After' 两列
# 并确保列顺序为 ['Before', 'After']
tmp = tmp.unstack('col')[['Before', 'After']]

# 6. 合并结果 (merge)
# 将计算出的 'Before' 和 'After' 列合并回原始DataFrame
# 使用 'Code' 列进行左连接,因为 'tmp' 的索引是 'Code'
out = df.merge(tmp, left_on='Code', right_index=True, how='left')

print("\n处理后的DataFrame:")
print(out)

输出结果:

原始DataFrame:
    Code  202001  202002  202003  202004  202005  202006  202007  202008  202009  202010  202011  202012    Date
0  12345    1000    1001    1002    1003    1004    1005    3006    1007    1008    1009    1010    1011  202004
1  12346     999    1000    1001    1002    1003    1003    1005    1006    1007    1008    1009    1010  202006
2  12347    1983    1984    1985    1986    1987    1988    1989    1990    1991    1992    1993    1994  202010

处理后的DataFrame:
    Code  202001  202002  202003  202004  202005  202006  202007  202008  202009  202010  202011  202012    Date  Before  After
0  12345    1000    1001    1002    1003    1004    1005    3006    1007    1008    1009    1010    1011  202004    3003  11063
1  12346     999    1000    1001    1002    1003    1003    1005    1006    1007    1008    1009    1010  202006    5005   7048
2  12347    1983    1984    1985    1986    1987    1988    1989    1990    1991    1992    1993    1994  202010   17883   5979

注意事项

  1. 数据类型一致性:在进行日期比较时,确保Date列和所有日期命名列(在melt后变为variable列)的数据类型一致且可比较。本例中,将它们都转换为字符串类型,利用字符串的字典序比较特性来判断日期先后。如果日期格式更复杂(如包含日),建议使用pd.to_datetime将其转换为datetime类型进行比较,这样更严谨。
  2. “之前”和“之后”的定义:本方案中,np.where(d['Date'].gt(d['variable']), 'Before', 'After')的逻辑是:如果Date(参考日期)严格大于variable(数据点日期),则算作“之前”;否则(即Date小于或等于variable),则算作“之后”。这意味着参考日期本身的数据点被包含在“之后”的总和中。如果需要不同的划分逻辑(例如,参考日期本身的数据点算作“之前”或单独处理),需要调整np.where的条件。
  3. 列名处理:如果原始DataFrame中已经存在名为Before或After的列,在执行此操作前,可能需要先将其删除(使用df.drop()),以避免冲突。
  4. 性能考虑:对于非常大的DataFrame,melt操作会显著增加DataFrame的行数。虽然Pandas的内部优化做得很好,但在极端情况下,应考虑其内存消耗。

总结

本教程提供了一种高效且灵活的方法,利用Pandas的melt、assign、groupby、unstack和merge等组合操作,实现了根据每行特定日期对DataFrame进行时间段划分并计算总和的功能。这种方法不仅代码简洁,而且逻辑清晰,能够很好地应对此类数据分析需求,极大地提高了数据处理的效率和灵活性。掌握这种数据重塑和聚合的技巧,对于进行复杂的时间序列数据分析至关重要。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
Python 时间序列分析与预测
Python 时间序列分析与预测

本专题专注讲解 Python 在时间序列数据处理与预测建模中的实战技巧,涵盖时间索引处理、周期性与趋势分解、平稳性检测、ARIMA/SARIMA 模型构建、预测误差评估,以及基于实际业务场景的时间序列项目实操,帮助学习者掌握从数据预处理到模型预测的完整时序分析能力。

49

2025.12.04

数据类型有哪几种
数据类型有哪几种

数据类型有整型、浮点型、字符型、字符串型、布尔型、数组、结构体和枚举等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.10.31

php数据类型
php数据类型

本专题整合了php数据类型相关内容,阅读专题下面的文章了解更多详细内容。

216

2025.10.31

mysql标识符无效错误怎么解决
mysql标识符无效错误怎么解决

mysql标识符无效错误的解决办法:1、检查标识符是否被其他表或数据库使用;2、检查标识符是否包含特殊字符;3、使用引号包裹标识符;4、使用反引号包裹标识符;5、检查MySQL的配置文件等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

179

2023.12.04

Python标识符有哪些
Python标识符有哪些

Python标识符有变量标识符、函数标识符、类标识符、模块标识符、下划线开头的标识符、双下划线开头、双下划线结尾的标识符、整型标识符、浮点型标识符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

271

2024.02.23

java标识符合集
java标识符合集

本专题整合了java标识符相关内容,想了解更多详细内容,请阅读下面的文章。

250

2025.06.11

c++标识符介绍
c++标识符介绍

本专题整合了c++标识符相关内容,阅读专题下面的文章了解更多详细内容。

121

2025.08.07

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

248

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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