
本文详细介绍了如何利用 pandas 库,根据另一个 dataframe 中定义的日期范围条件,对目标 dataframe 进行数据填充。通过将日期列转换为标准格式,结合 `melt`、`merge_asof` 和 `pivot` 等操作,实现高效且精确的条件性数据合并与筛选,最终生成符合特定日期区间要求的数据视图。
在数据分析和处理中,我们经常需要根据复杂的条件从一个数据集填充或筛选另一个数据集。其中一种常见场景是基于日期范围进行条件性数据填充。例如,我们有一个包含公司及其有效日期范围的 DataFrame (df1),以及另一个包含每日公司数据值的 DataFrame (df2)。我们的目标是创建一个新的 DataFrame (df3),其中 df2 中的值仅在对应公司的日期落在 df1 定义的有效范围内时才保留,否则置为 NaN。
数据准备
首先,我们定义两个示例 DataFrame df1 和 df2。df1 包含每个公司的有效起始和结束日期,而 df2 包含每日的各项公司数据。
import pandas as pd# 定义 df1:包含公司及其有效日期范围data1 = {'company': {0: 'a', 1: 'b', 2: 'c', 3: 'd'}, 'start date': {0: '2023-01-02', 1: '2023-01-05', 2: '2023-01-04', 3: '2023-01-03'}, 'end date': {0: '2023-01-06', 1: '2023-01-12', 2: '2023-01-13', 3: '2023-01-10'}}df1 = pd.DataFrame(data1)# 定义 df2:包含每日的公司数据data2 = {'DATE': {0: '2023-01-02', 1: '2023-01-03', 2: '2023-01-04', 3: '2023-01-05', 4: '2023-01-06', 5: '2023-01-09', 6: '2023-01-10', 7: '2023-01-11', 8: '2023-01-12', 9: '2023-01-13'}, 'a': {0: 1, 1: 2, 2: 3, 3: 4, 4: 5, 5: 6, 6: 7, 7: 8, 8: 9, 9: 10}, 'b': {0: 10, 1: 11, 2: 12, 3: 13, 4: 14, 5: 15, 6: 16, 7: 17, 8: 18, 9: 19}, 'c': {0: 30, 1: 31, 2: 32, 3: 33, 4: 34, 5: 35, 6: 36, 7: 37, 8: 38, 9: 39}, 'd': {0: 40, 1: 41, 2: 42, 3: 43, 4: 44, 5: 45, 6: 46, 7: 47, 8: 48, 9: 49}}df2 = pd.DataFrame(data2)print("df1 原始数据:")print(df1)print("ndf2 原始数据:")print(df2)
日期类型转换
在进行任何日期相关的操作之前,将日期列转换为 Pandas 的 datetime 类型至关重要。这确保了日期比较和合并的准确性。
df1['start date'] = pd.to_datetime(df1['start date'])df1['end date'] = pd.to_datetime(df1['end date'])df2['DATE'] = pd.to_datetime(df2['DATE'])print("ndf1 转换日期类型后:")print(df1)print("ndf2 转换日期类型后:")print(df2)
核心逻辑实现
为了实现基于日期范围的条件填充,我们将采用以下步骤:
数据重塑 (melt): 将 df2 从宽格式(每列代表一个公司)转换为长格式。这使得每个数据点(日期、公司、值)都成为一行,便于与 df1 进行合并。近似合并 (merge_asof): 使用 merge_asof 将 df2(长格式)与 df1 合并。merge_asof 是一种特殊的合并,它根据最近的键进行合并,而不是精确匹配。在这里,我们将 df2 的 DATE 与 df1 的 start date 进行近似合并,同时按 company 分组。条件筛选 (where): 合并后,我们将得到一个包含 DATE、company、value、start date 和 end date 的临时 DataFrame。然后,我们检查每个数据点的 DATE 是否落在对应的 start date 和 end date 之间。如果不在,则将 value 设置为 NaN。数据重塑 (pivot): 最后,将临时 DataFrame 重新 pivot 回原始的宽格式,以 DATE 为索引,company 为列,value 为值。
# 步骤1:将 df2 转换为长格式并按日期排序,以准备 merge_asoftmp = df2.melt('DATE', var_name='company').sort_values('DATE')# 步骤2:使用 merge_asof 进行近似合并# by='company' 表示在每个公司内部进行合并# left_on='DATE' 和 right_on='start date' 表示以 df2 的 DATE 和 df1 的 start date 进行近似匹配# 注意:merge_asof 要求左右 DataFrame 的合并键(这里是 DATE 和 start date)必须已排序df1_sorted = df1.sort_values('start date')tmp = pd.merge_asof(tmp, df1_sorted, by='company', left_on='DATE', right_on='start date')# 步骤3:根据日期范围条件筛选值# .where() 方法会根据条件保留值,不满足条件的置为 NaNdf3_filtered = tmp.assign(value=tmp['value'].where( (tmp['DATE'] >= tmp['start date']) & (tmp['DATE'] <= tmp['end date'])))# 步骤4:将数据透视回宽格式# index='DATE' 设置日期为行索引# columns='company' 设置公司为列名# values='value' 设置填充的值df3 = df3_filtered.pivot(index='DATE', columns='company', values='value') .rename_axis('', axis=1) .reset_index()print("ndf3 期望输出:")print(df3)
完整示例代码
以下是整合了所有步骤的完整代码,可以直接运行:
import pandas as pd# 原始数据定义data1 = {'company': {0: 'a', 1: 'b', 2: 'c', 3: 'd'}, 'start date': {0: '2023-01-02', 1: '2023-01-05', 2: '2023-01-04', 3: '2023-01-03'}, 'end date': {0: '2023-01-06', 1: '2023-01-12', 2: '2023-01-13', 3: '2023-01-10'}}df1 = pd.DataFrame(data1)data2 = {'DATE': {0: '2023-01-02', 1: '2023-01-03', 2: '2023-01-04', 3: '2023-01-05', 4: '2023-01-06', 5: '2023-01-09', 6: '2023-01-10', 7: '2023-01-11', 8: '2023-01-12', 9: '2023-01-13'}, 'a': {0: 1, 1: 2, 2: 3, 3: 4, 4: 5, 5: 6, 6: 7, 7: 8, 8: 9, 9: 10}, 'b': {0: 10, 1: 11, 2: 12, 3: 13, 4: 14, 5: 15, 6: 16, 7: 17, 8: 18, 9: 19}, 'c': {0: 30, 1: 31, 2: 32, 3: 33, 4: 34, 5: 35, 6: 36, 7: 37, 8: 38, 9: 39}, 'd': {0: 40, 1: 41, 2: 42, 3: 43, 4: 44, 5: 45, 6: 46, 7: 47, 8: 48, 9: 49}}df2 = pd.DataFrame(data2)# 1. 转换日期列为 datetime 类型df1['start date'] = pd.to_datetime(df1['start date'])df1['end date'] = pd.to_datetime(df1['end date'])df2['DATE'] = pd.to_datetime(df2['DATE'])# 2. 将 df2 转换为长格式并排序tmp = df2.melt('DATE', var_name='company').sort_values('DATE')# 3. 对 df1 进行排序,以满足 merge_asof 的要求df1_sorted = df1.sort_values('start date')# 4. 使用 merge_asof 进行近似合并# left_on='DATE' 和 right_on='start date' 确保将 df2 的 DATE 与 df1 中不大于该 DATE 的最近 start date 合并tmp = pd.merge_asof(tmp, df1_sorted, by='company', left_on='DATE', right_on='start date')# 5. 根据日期范围条件筛选值# 只有当 df2 的 DATE 在 df1 定义的 [start date, end date] 范围内时,才保留其值df3 = tmp.assign(value=tmp['value'].where( (tmp['DATE'] >= tmp['start date']) & (tmp['DATE'] <= tmp['end date']))) .pivot(index='DATE', columns='company', values='value') .rename_axis('', axis=1) .reset_index()print("n最终结果 df3:")print(df3)
注意事项与总结
日期类型的重要性: 始终确保涉及日期比较的列是 datetime 类型。不一致的类型会导致错误或意外结果。merge_asof 的排序要求: merge_asof 函数要求左右 DataFrame 的合并键(在本例中是 DATE 和 start date)必须是已排序的。如果未排序,结果将不准确。数据重塑的必要性: melt 和 pivot 操作是处理这种宽-长-宽转换问题的关键。它们使得在多列之间应用条件变得可行。条件逻辑: (tmp[‘DATE’] >= tmp[‘start date’]) & (tmp[‘DATE’] 性能考量: 对于非常大的数据集,melt 和 pivot 操作可能会消耗较多内存和计算资源。在处理大规模数据时,应评估其性能影响。
通过上述步骤,我们能够高效且准确地根据复杂的日期范围条件,从一个 DataFrame 中有选择性地填充另一个 DataFrame 的数据,这在财务分析、事件日志处理等领域具有广泛应用。
以上就是Pandas 数据框:基于日期区间条件的数据填充技巧的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1377439.html
微信扫一扫
支付宝扫一扫