使用 Pandas 高效处理分组数据:基于条件和日期排序创建新列

使用 pandas 高效处理分组数据:基于条件和日期排序创建新列

本文详细介绍了如何利用 Pandas 库处理复杂的分组数据操作。我们将学习如何结合 groupby、apply、sort_values、shift 和 cumsum 等方法,根据特定条件(如日期降序和数值变化)为 DataFrame 添加新列。教程将通过一个实际案例,演示如何高效地实现基于组内逻辑的条件累积计算,并确保结果正确对齐到原始数据结构。

引言

在数据分析和处理中,我们经常需要根据特定分组内的逻辑来生成新的数据列。这些逻辑可能涉及排序、条件判断以及累积计算。Pandas 提供了强大且灵活的工具集来应对此类挑战。本教程将以一个具体的案例为例,展示如何在一个 DataFrame 中,根据 text 列进行分组,然后根据 date 列的降序以及 number 列的数值变化,计算并添加一个名为 test 的新列。

问题描述

假设我们有以下 Pandas DataFrame:

import pandas as pdimport numpy as npdata = {    'id': [1, 2, 3, 4, 5, 6, 7],    'date': ['2019-02-01', '2019-02-10', '2019-02-25', '2019-03-05', '2019-03-16', '2019-04-05', '2019-05-15'],    'date_difference': [None, 9, 15, 11, 10, 19, 40],    'number': [1, 0, 1, 0, 0, 0, 0],    'text': ['A', 'A', 'A', 'A', 'A', 'B', 'B']}df = pd.DataFrame(data)print("原始 DataFrame:")print(df)

原始 DataFrame 如下所示:

id date date_difference number text

12019-02-01NULL1A22019-02-1090A32019-02-25151A42019-03-05110A52019-03-16100A62019-04-05190B72019-05-15400B

我们的目标是根据 text 列进行分组,并在每个组内,依据 date 列的降序,生成一个名为 test 的新列。生成 test 列的规则如下:

在每个组内,从日期降序排列的第一个条目开始计算。当 number 列的值为 0 时,步长(step size)初始为 1。当遇到 number 列的值为 1 时,步长增加 1。如果组内 number 列中没有 1,则整个组的步长始终保持为 1。

期望的最终 DataFrame 如下:

id date date_difference number text test

12019-02-01NULL1A222019-02-1090A232019-02-25151A142019-03-05110A152019-03-16100A162019-04-05190B172019-05-15400B1

解决方案实现

解决此问题的关键在于正确地结合 Pandas 的分组、排序、位移和累积求和操作。

核心思路

分组 (Group by): 首先,我们需要根据 text 列对 DataFrame 进行分组,因为 test 列的计算逻辑是针对每个 text 组独立的。组内排序 (Sort within group): 问题的关键在于“从日期降序开始计算”。这意味着在每个组内部,我们需要先按 date 列降序排列数据,然后进行计算。位移 (Shift): 为了实现“当 number == 0 时步长为 1,当找到 1 时步长增加 1”的逻辑,我们可以将 number 列进行位移操作。将 number 列向前位移一位,并用 1 填充位移后产生的第一个缺失值。这样,对于原始 number 为 0 的行,其对应的位移后值通常是其下一个条目的 number 值;而对于原始 number 为 1 的行,其对应的位移后值会影响其之前的行的计算。fill_value=1 确保了序列的起始值(即日期降序的第一个值)为 1,符合“步长初始为 1”的条件。累积求和 (Cumulative Sum): 对位移后的 number 列进行累积求和,即可得到所需的 test 值。这个累积和会根据 number 列中的 1 进行递增。结果对齐 (Align Results): groupby().apply() 返回的结果通常会保留原始 DataFrame 的索引,但如果内部进行了排序,则需要确保最终结果能正确地与原始 DataFrame 对齐。assign() 方法在这里非常有用,它能自动根据索引将新生成的 Series 与原 DataFrame 合并。

完整代码

import pandas as pdimport numpy as npdata = {    'id': [1, 2, 3, 4, 5, 6, 7],    'date': ['2019-02-01', '2019-02-10', '2019-02-25', '2019-03-05', '2019-03-16', '2019-04-05', '2019-05-15'],    'date_difference': [None, 9, 15, 11, 10, 19, 40],    'number': [1, 0, 1, 0, 0, 0, 0],    'text': ['A', 'A', 'A', 'A', 'A', 'B', 'B']}df = pd.DataFrame(data)# 将 'date' 列转换为 datetime 类型,以便正确排序df['date'] = pd.to_datetime(df['date'])# 使用 assign 方法添加新列 'test'df_result = df.assign(    test=df    # 1. 按 'text' 列进行分组    .groupby("text")    # 2. 对每个组应用一个函数    .apply(        lambda g: (            # 3. 在组内按 'date' 列降序排序            g.sort_values(by="date", ascending=False)            # 4. 对 'number' 列进行位移,向前一位,并用 1 填充缺失值            .number.shift(periods=1, fill_value=1)            # 5. 对位移后的结果进行累积求和            .cumsum()        )    )    # 6. 移除 apply 产生的 'text' 索引层,使 Series 索引与原始 df 索引一致    .droplevel("text")    # assign 方法会自动将结果 Series 与原始 DataFrame 的索引对齐)print("n最终 DataFrame:")print(df_result)

逻辑解析与示例跟踪

让我们以 text 为 ‘A’ 的组为例,详细解释每一步的操作:

原始 text=’A’ 的数据(按原始索引顺序):

id date number

12019-02-01122019-02-10032019-02-25142019-03-05052019-03-160

g.sort_values(by=”date”, ascending=False):将组内数据按 date 降序排列。排序后的数据(索引为原始 DataFrame 索引):

id date number

52019-03-16042019-03-05032019-02-25122019-02-10012019-02-011

.number.shift(periods=1, fill_value=1):对排序后的 number 列 [0, 0, 1, 0, 1] 进行向前位移,并用 1 填充第一个位置。位移后的 Series: [1, 0, 0, 1, 0] (对应索引 [5, 4, 3, 2, 1])

.cumsum():对位移后的 Series [1, 0, 0, 1, 0] 进行累积求和。累积和结果: [1, 1, 1, 2, 2] (对应索引 [5, 4, 3, 2, 1])

这意味着:

df.loc[5, ‘test’] = 1df.loc[4, ‘test’] = 1df.loc[3, ‘test’] = 1df.loc[2, ‘test’] = 2df.loc[1, ‘test’] = 2

.droplevel(“text”):apply 方法在返回 Series 时,如果 groupby 包含多个键或 apply 的结果不是单个 Series,可能会产生 MultiIndex。在这里,apply 内部返回的是一个 Series,其索引是原始 DataFrame 的索引,但由于 groupby(“text”),其上会有一个 text 层的 MultiIndex。droplevel(“text”) 移除了这个额外的索引层,使得最终 Series 的索引与原始 df 的索引完全匹配。

df.assign(test=…):assign 方法将这个计算好的 Series 作为 test 列添加到原始 df 中。Pandas 会自动根据索引进行对齐,确保 test 值回到其原始的行位置。

最终结果与期望输出完全一致。对于 text=’B’ 的组,由于 number 列中没有 1,shift(fill_value=1) 会使所有值变为 1,cumsum() 结果也都是 1,符合“没有 1 时步长保持为 1”的规则。

注意事项与总结

日期类型转换: 在进行日期排序之前,确保 date 列的数据类型是 datetime。如果不是,需要使用 pd.to_datetime() 进行转换。groupby().apply() 的灵活性: apply() 方法非常强大,允许你在每个分组上执行几乎任何自定义操作。但需要注意其性能可能不如优化的 Pandas 方法(如 transform 或 agg),对于大型数据集应谨慎使用。然而,对于这种复杂的、需要组内排序和

以上就是使用 Pandas 高效处理分组数据:基于条件和日期排序创建新列的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1366673.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月14日 06:46:41
下一篇 2025年12月14日 06:46:56

相关推荐

  • 基于Pandas的Groupby操作添加条件列的教程

    本文详细介绍了如何使用Pandas的groupby操作,并结合条件判断,向DataFrame中添加新的列。通过示例代码,展示了如何根据分组内的特定条件,计算并生成新的列值,尤其是在需要考虑组内顺序和累计效应时,提供了一种高效的解决方案。 在数据分析中,经常需要在DataFrame中基于分组信息和特定…

    好文分享 2025年12月14日
    000
  • 基于分组和条件添加新列:Pandas教程

    本文详细介绍了如何使用 Pandas 在 DataFrame 中基于分组和条件添加新列。通过 groupby()、apply()、sort_values()、shift() 和 cumsum() 等函数的组合使用,可以实现复杂的数据转换和列生成。本文提供清晰的代码示例和详细的步骤解释,帮助读者理解并…

    2025年12月14日
    000
  • 基于分组和条件判断添加新列:Pandas 教程

    本文旨在讲解如何使用 Pandas 在数据框中基于分组和条件判断来创建新的列。通过 groupby()、apply()、sort_values()、shift() 和 cumsum() 等函数,可以实现复杂的数据转换和计算,从而生成符合特定业务逻辑的新列。文章提供详细的代码示例和步骤解释,帮助读者理…

    2025年12月14日
    000
  • SymPy表达式在终端与GUI中的美观显示方法

    本教程旨在解决在Python环境中,尤其是在Pydroid3终端和Tkinter GUI中,如何美观地显示SymPy数学表达式的问题。文章将深入探讨SymPy库提供的pprint()和pretty()函数,它们能够生成易于阅读的文本格式表达式。通过具体的代码示例,教程将展示如何在不同场景下利用这些函…

    2025年12月14日
    000
  • SymPy表达式在Pydroid3终端与GUI中的美观显示策略

    本文探讨了在Pydroid3终端以及GUI环境中美观显示SymPy数学表达式的方法。针对init_printing在特定环境下可能失效的问题,详细介绍了如何利用sympy.pprint和sympy.pretty函数生成字符画形式的表达式,并探讨了在Tkinter等GUI界面中显示这些表达式的策略,以…

    2025年12月14日
    000
  • SymPy表达式在Pydroid3终端与GUI中的美观显示方法

    本文旨在解决在Pydroid3等移动开发环境中,SymPy表达式无法正常美观显示的问题。传统init_printing方法可能失效,但可通过sympy.pprint()或sympy.pretty()函数获取格式化字符串,从而在终端中实现美观输出。对于GUI显示,将探讨将这些字符串集成到Tkinter…

    2025年12月14日
    000
  • 在Pydroid3中美观打印SymPy表达式及GUI显示方案

    本教程旨在解决在Pydroid3环境中美观打印SymPy表达式的问题,特别是当init_printing无效时。文章将详细介绍如何利用SymPy内置的pprint()和pretty()函数在终端输出格式化的数学表达式。同时,针对在Tkinter或其他GUI框架中显示复杂数学表达式的需求,本教程将探讨…

    2025年12月14日
    000
  • Django Update 语句未按预期更新数据库

    Django Update 语句未按预期更新数据库 本文旨在解决 django 框架中使用 update() 方法更新数据库时遇到的问题,特别是在条件判断后更新数据但数据库未按预期更改的情况。我们将深入探讨 update() 方法的特性,并提供解决方案,确保数据更新的正确性和一致性。 在使用 Dja…

    2025年12月14日
    000
  • 解决 Scikit-learn FeatureUnion 陷入死循环的问题

    本文旨在解决在使用 Scikit-learn 的 FeatureUnion 时遇到的无限循环问题。通过分析问题代码,明确了 FeatureUnion 并行执行的特性,并解释了并行执行导致资源过度消耗的原因,最终提供了避免此类问题的解决方案,帮助读者更有效地利用 FeatureUnion 进行特征工程…

    2025年12月14日
    000
  • 解决Scikit-learn FeatureUnion卡死问题

    问题背景与解决方案 在使用Scikit-learn的FeatureUnion进行特征工程时,有时会遇到程序长时间运行甚至卡死的情况,尤其是在结合RFE(Recursive Feature Elimination)等计算密集型算法时。这往往是因为对FeatureUnion的并行执行机制理解不足导致的。…

    2025年12月14日
    000
  • Python怎样实现数据滑动窗口?rolling计算

    处理滑动窗口中的缺失值可通过设置min_periods参数确保窗口内至少有指定数量的非缺失值参与计算,或在自定义函数中使用dropna()忽略nan值;2. 滑动窗口在时间序列分析中用于趋势分析、季节性检测、异常值识别和预测建模;3. 除pandas的rolling()外,还可使用numpy的con…

    2025年12月14日
    000
  • Python如何实现代码依赖分析?importlib检测

    传统的静态分析工具无法完全满足python依赖检测,因为它们仅扫描import语句,无法处理运行时动态导入(如__import__、条件导入、exec执行的代码)以及c扩展的隐式依赖;2. 利用importlib的导入钩子(import hooks)进行运行时依赖追踪,可通过自定义metapathf…

    2025年12月14日
    000
  • Python如何实现基于拓扑数据分析的异常模式发现?

    基于拓扑数据分析(tda)的异常模式发现,通过提取数据的拓扑结构特征实现异常识别。1. 数据预处理阶段将原始数据转换为点云或距离矩阵;2. 使用gudhi或ripser库计算持久同源性,生成持久图以捕捉数据的连通性与“洞”的生命周期;3. 将持久图转化为固定长度的特征向量,常用方法包括持久图图像、持…

    2025年12月14日 好文分享
    000
  • 如何用Python源码构建影视素材库 Python源码支持分类与检索功能

    核心答案是通过python脚本自动化扫描文件、提取元数据并存入sqlite数据库实现分类与检索;2. 具体步骤为:先用os模块遍历目录解析文件名获取标题等信息,结合moviepy或ffprobe提取时长等数据;3. 设计数据库时创建media_items主表及genres、tags独立表并通过关联表…

    2025年12月14日 好文分享
    000
  • Python如何实现自动化测试?Selenium教程

    搭建selenium自动化测试环境步骤如下:1.安装python并配置环境变量;2.确保pip已安装;3.使用pip安装selenium库;4.安装webdriver_manager库以自动管理浏览器驱动;5.安装目标浏览器如chrome。使用selenium进行元素交互和断言的方法包括:通过id、…

    2025年12月14日 好文分享
    000
  • Django登录失败后Alert消息不显示的调试与修复

    本文旨在解决Django用户登录验证失败后,前端Alert消息未能正确显示的问题。通过检查HTML模板中的JavaScript代码拼写错误,以及Django视图函数中的渲染逻辑,提供修复方案,确保用户在登录失败时能收到清晰的错误提示,从而提升用户体验。 在Django开发中,用户登录失败后显示错误提…

    2025年12月14日
    000
  • 如何用Python发现未初始化的变量使用?

    python中“未初始化变量”问题实质是名字未绑定导致的nameerror,解决方法主要有两条路径:一是使用静态代码分析工具(如pylint、flake8)在运行前发现潜在问题;二是通过运行时异常处理和调试工具捕获错误。静态分析工具通过解析ast检查代码结构,提前预警未定义变量使用;运行时则可使用t…

    2025年12月14日 好文分享
    000
  • 如何使用Python发现不安全的字符串格式化?

    python中发现不安全字符串格式化的最直接方法是使用静态代码分析工具如bandit,1.集成bandit等工具到开发流程中自动识别漏洞;2.通过人工审查关注外部输入与格式化结合的逻辑;3.编写包含恶意输入的测试用例验证安全性。常见陷阱包括注入攻击、日志注入和任意代码执行,核心在于信任未经处理的输入…

    2025年12月14日 好文分享
    000
  • Python如何调试代码?快速定位错误方法

    调试python代码的核心在于选择合适的工具和方法。1.使用print语句可在小型脚本中快速查看变量和执行流程;2.使用pdb调试器可逐行执行代码、查看变量并设置断点;3.使用ide(如vs code、pycharm)可图形化调试,提升效率;4.处理异常通过try…except结构防止程…

    2025年12月14日 好文分享
    000
  • Python源码中如何实现模块缓存机制 解析importlib的缓存处理逻辑

    python模块缓存机制通过sys.modules字典实现,确保模块只被加载一次。1. 导入时,解释器首先检查sys.modules,若存在则直接返回模块对象;2. 若不存在,则通过importlib执行查找、加载、执行三步流程;3. 模块执行前,空模块对象即被放入sys.modules,形成“先占…

    2025年12月14日 好文分享
    000

发表回复

登录后才能评论
关注微信