新闻中心

Pandas高效数据处理:利用bisect优化条件查找最新匹配索引

2025-11-06
浏览次数:
返回列表

Pandas高效数据处理:利用bisect优化条件查找最新匹配索引

本文探讨了在pandas dataframe中高效查找满足特定条件的最新历史索引的方法。针对传统`df.apply`方法的性能瓶颈,文章详细介绍了基于python内置`bisect`模块的优化方案。通过对比多种实现,重点展示了`bisect`在处理大规模数据集时显著的性能优势,并提供了详细的代码示例与解释,旨在帮助读者提升pandas数据处理效率。

在数据分析和处理中,我们经常会遇到需要基于当前行数据,从历史数据中查找满足特定条件的记录,并获取其相关信息(例如索引或日期)的场景。一个典型的例子是:给定一个DataFrame,其中包含“lower”和“upper”两列以及一个日期索引,我们需要为每一行找到其之前所有行中,“lower”值大于等于当前行“upper”值的最新记录的日期。

1. 问题描述与低效实现

考虑以下Pandas DataFrame示例,其中包含lower和upper两列,并以日期作为索引:

import pandas as pd
import numpy as np

# 示例 DataFrame
data = {'lower': [7, 1, 6, 1, 1, 1, 1, 11, 1, 1],
        'upper': [2, 3, 4, 5, 6, 7, 8, 9, 10, 11]}

df = pd.DataFrame(data=data)
df['DATE'] = pd.date_range('2025-01-01', periods=len(data['lower']))
df.set_index('DATE', inplace=True)

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

输出:

原始DataFrame:
            lower  upper
DATE                    
2025-01-01      7      2
2025-01-02      1      3
2025-01-03      6      4
2025-01-04      1      5
2025-01-05      1      6
2025-01-06      1      7
2025-01-07      1      8
2025-01-08     11      9
2025-01-09      1     10
2025-01-10      1     11

我们的目标是添加一个新列prev,其中包含满足条件previous_row['lower'] >= current_row['upper']的最新历史记录的DATE。

一种直观但效率低下的实现方式是使用df.apply()结合df.loc进行行迭代:

def get_most_recent_index_baseline(row, dataframe):
    # 查找当前行之前的行
    # 注意:row.name - pd.Timedelta(minutes=1) 假设索引是分钟频率,确保不包含当前行
    previous_indices = dataframe.loc[:row.name - pd.Timedelta(minutes=1)]  
    # 在之前的行中筛选满足条件的记录,并获取其最大(即最新)索引
    recent_index = previous_indices[previous_indices['lower'] >= row['upper']].index.max()
    return recent_index

# 应用函数
# df['prev'] = df.apply(lambda row: get_most_recent_index_baseline(row, df), axis=1)
# print("\n使用df.apply()的结果:")
# print(df)

这种方法的问题在于,df.apply(axis=1)本质上是逐行迭代,并且在每次迭代中,previous_indices = dataframe.loc[:row.name - pd.Timedelta(minutes=1)]会创建一个DataFrame的切片副本,然后进行条件筛选和max()操作。对于大型DataFrame,这种重复的切片和操作会导致极高的计算开销,性能非常差。

2. 性能瓶颈分析与优化需求

为了量化性能差异,我们通常会使用一个更大的数据集进行基准测试。以下是一个生成测试数据的函数和基准测试的设置:

def get_sample_df(rows=100_000):
    # Sample DataFrame
    data = {'lower': np.random.default_rng(seed=1).uniform(1,100,rows),
            'upper': np.random.default_rng(seed=2).uniform(1,100,rows)}

    df = pd.DataFrame(data=data)
    df = df.astype(int)

    df['DATE'] = pd.date_range('2025-01-01', periods=len(data['lower']), freq="min")
    df.set_index('DATE', inplace=True)
    return df

# get_baseline 函数封装了上述df.apply()的逻辑
def get_baseline():
    df = get_sample_df()
    def get_most_recent_index(row):
        previous_indices = df.loc[:row.name - pd.Timedelta(minutes=1)]  
        recent_index = previous_indices[previous_indices['lower'] >= row['upper']].index.max()
        return recent_index
    df['prev'] = df.apply(get_most_recent_index, axis=1) 
    return df

# 基准测试结果 (针对100,000行数据)
# baseline: 1min 35s ± 5.15 s per loop (mean ± std. dev. of 2 runs, 2 loops each)

从基准测试结果可以看出,对于10万行数据,df.apply()方法需要大约1分35秒,这在实际应用中是不可接受的。因此,我们需要寻找更高效的算法来解决这个问题。

3. 基于二分查找 (bisect) 的高效优化方案

由于问题涉及到依赖于过去状态的查找,完全的Pandas矢量化操作可能难以直接实现。然而,我们可以通过结合Python的bisect模块和自定义迭代逻辑来大幅提升性能。bisect模块实现了二分查找算法,可以在有序序列中高效地查找元素插入点。

核心思想是:

  1. 维护一个已见过的lower值及其对应的最新日期(last_seen字典)。
  2. 为了快速找到所有大于等于当前upper值的lower值,我们预先对所有唯一的lower值进行排序(uniq_lower)。
  3. 对于每一行,使用bisect_left在uniq_lower中找到第一个大于等于当前upper值的位置。
  4. 从该位置开始,遍历uniq_lower中所有满足条件的lower值,并在last_seen字典中查找它们的最新日期,取其中最大的日期作为结果。
  5. 处理完当前行后,更新last_seen字典中当前行lower值对应的日期。

以下是使用bisect实现的优化方案:

刺鸟创客 刺鸟创客

一款专业高效稳定的AI内容创作平台

刺鸟创客 110 查看详情 刺鸟创客
from bisect import bisect_left

def get_bisect():
    df = get_sample_df() # 使用与基准测试相同的样本数据

    def get_prev_bs(lower_series, upper_series, date_index):
        # 获取所有唯一的lower值并排序,用于二分查找
        uniq_lower = sorted(set(lower_series))
        # 存储每个lower值最近出现的日期
        last_seen = {}

        # 迭代DataFrame的每一行
        for l, u, d in zip(lower_series, upper_series, date_index):
            # 使用bisect_left找到在uniq_lower中第一个大于等于u的元素的索引
            # 这意味着uniq_lower[idx:]包含了所有可能满足条件 lower >= u 的值
            idx = bisect_left(uniq_lower, u)

            max_date = None
            # 遍历所有满足条件的lower值
            for lv in uniq_lower[idx:]:
                # 如果这个lower值之前出现过
                if lv in last_seen:
                    # 更新max_date为最近的日期
                    if max_date is None:
                        max_date = last_seen[lv]
                    elif last_seen[lv] > max_date:
                        max_date = last_seen[lv]

            # 返回当前行的结果
            yield max_date

            # 更新last_seen字典:当前lower值l的最新日期是d
            last_seen[l] = d

    df["prev"] = list(get_prev_bs(df["lower"], df["upper"], df.index))
    return df

# 基准测试结果 (针对100,000行数据)
# bisect: 1.76 s ± 82.5 ms per loop (mean ± std. dev. of 2 runs, 2 loops each)

# 打印使用bisect方法的示例结果 (使用小规模df)
def get_bisect_example():
    data = {'lower': [7, 1, 6, 1, 1, 1, 1, 11, 1, 1],
            'upper': [2, 3, 4, 5, 6, 7, 8, 9, 10, 11]}
    df = pd.DataFrame(data=data)
    df['DATE'] = pd.date_range('2025-01-01', periods=len(data['lower']))
    df.set_index('DATE', inplace=True)

    def get_prev_bs_example(lower_series, upper_series, date_index):
        uniq_lower = sorted(set(lower_series))
        last_seen = {}
        for l, u, d in zip(lower_series, upper_series, date_index):
            idx = bisect_left(uniq_lower, u)
            max_date = None
            for lv in uniq_lower[idx:]:
                if lv in last_seen:
                    if max_date is None:
                        max_date = last_seen[lv]
                    elif last_seen[lv] > max_date:
                        max_date = last_seen[lv]
            yield max_date
            last_seen[l] = d
    df["prev"] = list(get_prev_bs_example(df["lower"], df["upper"], df.index))
    return df

print("\n使用bisect()方法的结果:")
print(get_bisect_example())

输出:

使用bisect()方法的结果:
            lower  upper       prev
DATE                           
2025-01-01      7      2        NaT
2025-01-02      1      3 2025-01-01
2025-01-03      6      4 2025-01-01
2025-01-04      1      5 2025-01-03
2025-01-05      1      6 2025-01-03
2025-01-06      1      7 2025-01-01
2025-01-07      1      8        NaT
2025-01-08     11      9        NaT
2025-01-09      1     10 2025-01-08
2025-01-10      1     11 2025-01-08

从基准测试结果来看,bisect方法将处理10万行数据的时间从1分35秒大幅缩短至约1.76秒,性能提升了近50倍,这在处理大规模数据集时是决定性的优势。

4. 其他尝试及考量

在解决这类问题时,通常会有多种尝试,但并非所有都适用于所有场景:

4.1 pyjanitor库的尝试

pyjanitor是一个提供整洁API的Pandas扩展库,其中包含conditional_join等高级功能,可以用于执行条件连接操作。理论上,它可能提供一种矢量化的解决方案。

# def get_pyjanitor():
#     df = get_sample_df()
#     df.reset_index(inplace=True)
#     left_df = df.assign(index_prev=df.index)
#     right_df = df.assign(index_next=df.index)
#     out=(left_df
#         .conditional_join(
#             right_df, 
#             ('lower','upper','>='), 
#             ('index_prev','index_next','<'), 
#             df_columns='index_prev', 
#             right_columns=['index_next','lower','upper'])
#         )
#     # 后续处理逻辑以找到最近的匹配项
#     # ...
#     return df

# 基准测试结果:
# Unable to allocate 37.2 GiB for an array with shape (4994299505,) and data type int64

尽管pyjanitor提供了一种结构化的方法,但在实际测试中,对于中等规模以上的数据集(例如10万行),conditional_join操作可能会导致巨大的内存分配错误(如上述Unable to allocate 37.2 GiB),使其在内存受限的环境下不可用。这是因为条件连接可能产生一个非常大的中间结果集。

4.2 基于enumerate的直接迭代

另一种尝试是使用enumerate和列表操作进行迭代。这种方法与df.apply类似,也是逐行处理,但可能通过直接操作Python列表来避免Pandas内部的一些开销。

# def get_enumerate():
#     df = get_sample_df()
#     df.reset_index(inplace=True)

#     date_list=df["DATE"].values.tolist()
#     lower_list=df["lower"].values.tolist()
#     upper_list=df["upper"].values.tolist()
#     new_list=[]
#     for i,(x,y) in enumerate(zip(lower_list,upper_list)):
#         if i==0:
#             new_list.append(None)
#         else:
#             if (any(j >= y for j in lower_list[0:i])): # 检查是否存在满足条件的项
#                 for ll,dl in zip(reversed(lower_list[0:i]),reversed(date_list[0:i])): # 从后往前查找最近的
#                     if ll>=y:
#                         new_list.append(dl)
#                         break
#             else:
#                 new_list.append(None)
#     df['prev']=new_list
#     df['prev']=pd.to_datetime(df['prev'])
#     return df

# 基准测试结果:
# enumerate: 1min 13s ± 2.17 s per loop (mean ± std. dev. of 2 runs, 2 loops each)

这种方法虽然比df.apply()略快,但其时间复杂度仍然较高,因为它在每次迭代中都可能需要遍历之前的所有元素(reversed(lower_list[0:i])),导致整体性能依然不佳,对于大规模数据仍无法满足要求。

5. 总结与注意事项

本文详细探讨了在Pandas DataFrame中查找满足特定条件的最新历史索引的问题,并对比了多种实现方法的性能。

  • df.apply()方法虽然直观易懂,但由于其逐行迭代和重复的DataFrame切片操作,导致性能极差,不适用于大数据集。
  • pyjanitor的conditional_join在理论上具有矢量化潜力,但在实际应用中,对于数据量稍大的情况,可能因内存消耗过大而无法使用。
  • 基于enumerate的直接迭代相比df.apply()略有改进,但本质上仍是O(N^2)的复杂度,性能瓶颈依然存在。
  • 基于Python bisect模块的优化方案是目前最有效的解决方案。通过结合二分查找和字典来维护历史状态,它将时间复杂度显著降低,从而在处理10万行数据时实现了从分钟级别到秒级别的性能飞跃。

注意事项:

  1. 问题特性决定优化方向: 本文的问题具有“依赖过去状态”的特性,这使得完全的Pandas矢量化(如NumPy操作)难以直接应用。在这种情况下,结合高效的算法(如二分查找)和Python原生数据结构(如字典、列表)进行优化是关键。
  2. 数据结构选择: last_seen字典能够以O(1)的平均时间复杂度查找特定lower值对应的最新日期,这对于性能至关重要。
  3. 预排序的优势: uniq_lower列表的预排序结合bisect_left,使得在查找所有满足lv >= u条件的lower值时非常高效。
  4. 内存与时间权衡: 在选择优化方案时,需要综合考虑时间和内存开销。例如,pyjanitor可能在某些场景下很快,但其内存消耗可能成为瓶颈。

总之,在Pandas数据处理中遇到涉及历史状态依赖的复杂条件查找时,深入理解问题特性,并灵活运用Python的内置高效算法(如bisect),是实现高性能数据处理的关键。

以上就是Pandas高效数据处理:利用bisect优化条件查找最新匹配索引的详细内容,更多请关注其它相关文章!


# 大数据  # app  # 性能瓶颈  # 迭代  # 数据处理  # 数据结构  # 行数  # 遍历  # python  # 矢量化  # 考拉海购的推广营销  # seo过时的技巧 案例  # seo策略完整版  # 资深的口碑营销推广  # 朔州关键词排名系统  # 湖北网站优化引流  # 宁夏互联网营销推广渠道  # 门窗seo优化软件  # 但在  # 第一个  # 其中包含  # 是一个  # 郑州网站推广公司电话  # 广西哪里找网络营销推广 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: 小红书商家版怎样在笔记嵌入商品卡路径_小红书商家版在笔记嵌入商品卡路径【挂载教程】  在J*a中如何开发在线活动报名与管理系统_活动报名管理项目实战解析  qq邮箱日历功能怎么用_创建日程与会议邀请的技巧  Descript怎样用AI剪辑自动去噪_Descript用AI剪辑自动去噪【自动降噪】  《马克思佩恩3》早期版本曝光 UI设计曾多次调整!  Win10系统怎么查看已安装更新_Win10卸载有问题的更新补丁  虚幻5科幻题材ARPG大作遭取消!本是《奇异人生》厂商新作  韩剧圈正版入口页面_韩剧圈官网登录链接  CSS自定义字体样式被系统字体替换怎么办_font-face方式指定font-display控制渲染策略  MAC如何安全彻底地删除文件_MAC使用终端命令确保文件无法被恢复  2026春节假期票务安排_2026春节放假购票指南  Yandex免登录官网入口_俄罗斯Yandex搜索引擎直达链接  poki网页游戏推荐_poki免费游戏平台入口  Composer的 archive 命令怎么用_快速打包你的PHP项目及其Composer依赖  支付宝碰一碰设备是REDMI手机吗 博主拆机辟谣:处理器、内存都不一样  双系统安装时,如何设置默认启动系统? msconfig命令了解一下!  css绝对定位元素脱离父容器怎么办_确保父元素position非static  C++20的source_location是什么_C++在编译期获取源码位置信息用于日志和断言  b站怎么删除评论_b站评论管理与删除操作  Safari怎么安装扩展程序 浏览器插件安装与管理方法【详解】  Bing引擎入口最新2025 Bing搜索免费官方登录  2306选座时如何选靠窗位置_12306选座靠窗座位查看方法解析  Golang如何使用buffered channel提高性能_Golang buffered channel优化技巧  Windows10怎么开启夜间模式 Windows10系统设置调整色温与亮度缓解夜间用眼疲劳【教程】  电脑IP地址怎么查 查看本机IP地址的几种方法  sublime如何优雅地处理行尾空格_sublime自动清理多余空白字符配置  c++ 获取系统当前时间 c++时间戳获取方法  在哪找SublimeJ远程工具_SFTP插件配置教程  精准捕获:如何在页面中监听除特定元素外的所有点击事件  单12V-2&#215;6实现为RTX 5090供电750W!甚至都没敢跑分  PostgreSQL海量数据高效导入策略:Python与Django实践指南  从OpenAI API响应中高效提取生成文本  如何为你的Composer包编写自动化测试_集成PHPUnit到Composer的scripts工作流  漫蛙漫画官方首页 漫蛙2漫画在线阅读入口  Win11网速慢怎么解决 Win11网络设置优化解除限速  PHP中获取MongoDB服务器运行时间(Uptime)的专业指南  飞书妙记怎样用语音转文字速记_飞书妙记用语音转文字速记【速记方法】  漫蛙漫画官方主页入口 漫蛙MANWA网页直达访问链接  解决Rails应用中内容错位与Turbo警告:meta标签误用导致富文本渲染异常  海棠电脑版入口_通过电脑访问海棠官网阅读  Win11怎么关闭触摸屏_Windows 11禁用HID符合标准触摸屏  win11 arm版怎么安装 M1/M2 Mac虚拟机安装ARM win11的方法  三星ZFold5多任务卡顿_Samsung ZFold5流畅度提升  如何使用Node.js csv 包按条件移除含空字段的CSV记录  Go语言中JSON数据解析与字段访问教程  如何使 Jest 模拟函数默认抛出错误以提高测试效率  动漫花园资源网使用步骤_动漫花园资源网下载流程  J*a应用程序首次运行自动创建文件与目录的最佳实践  Yandex搜索引擎一键访问入口_俄罗斯Yandex官网免登录  汽水音乐网页版使用入口_汽水音乐电脑版播放指南 

搜索