python - 如何计算 Pandas 中重叠的日期时间间隔?

标签 python pandas datetime count

我有一个包含两个日期时间列的以下 DataFrame:

    start               end
0   01.01.2018 00:47    01.01.2018 00:54
1   01.01.2018 00:52    01.01.2018 01:03
2   01.01.2018 00:55    01.01.2018 00:59
3   01.01.2018 00:57    01.01.2018 01:16
4   01.01.2018 01:00    01.01.2018 01:12
5   01.01.2018 01:07    01.01.2018 01:24
6   01.01.2018 01:33    01.01.2018 01:38
7   01.01.2018 01:34    01.01.2018 01:47
8   01.01.2018 01:37    01.01.2018 01:41
9   01.01.2018 01:38    01.01.2018 01:41
10  01.01.2018 01:39    01.01.2018 01:55

我想计算在给定时间结束之前有多少开始(间隔)同时处于事件状态(换句话说:每行与其余行重叠多少次行数)。

例如从 00:47 到 00:52 只有一个处于事件状态,从 00:52 到 00:54 有两个处于事件状态,从 00:54 到 00:55 又只有一个处于事件状态,依此类推。

我尝试将列堆叠在一起,按日期排序,并通过迭代整个数据帧,给每个“开始”+1 来计数器,给每个“结束”-1。它可以工作,但在我的原始数据框架上,我有几百万行,迭代需要永远 - 我需要找到一种更快的方法。

我原来的基本但不是很好代码:

import pandas as pd
import numpy as np

df = pd.read_csv('something.csv', sep=';')

df = df.stack().to_frame()
df = df.reset_index(level=1)
df.columns = ['status', 'time']
df = df.sort_values('time')
df['counter'] = np.nan
df = df.reset_index().drop('index', axis=1)

print(df.head(10))

给出:

    status  time                counter
0   start   01.01.2018 00:47    NaN
1   start   01.01.2018 00:52    NaN
2   stop    01.01.2018 00:54    NaN
3   start   01.01.2018 00:55    NaN
4   start   01.01.2018 00:57    NaN
5   stop    01.01.2018 00:59    NaN
6   start   01.01.2018 01:00    NaN
7   stop    01.01.2018 01:03    NaN
8   start   01.01.2018 01:07    NaN
9   stop    01.01.2018 01:12    NaN

和:

counter = 0

for index, row in df.iterrows():

    if row['status'] == 'start':
        counter += 1
    else:
        counter -= 1
    df.loc[index, 'counter'] = counter

最终输出:

        status  time                counter
    0   start   01.01.2018 00:47    1.0
    1   start   01.01.2018 00:52    2.0
    2   stop    01.01.2018 00:54    1.0
    3   start   01.01.2018 00:55    2.0
    4   start   01.01.2018 00:57    3.0
    5   stop    01.01.2018 00:59    2.0
    6   start   01.01.2018 01:00    3.0
    7   stop    01.01.2018 01:03    2.0
    8   start   01.01.2018 01:07    3.0
    9   stop    01.01.2018 01:12    2.0

有什么方法可以通过使用 iterrows() 来做到这一点吗?

提前致谢!

最佳答案

使用Series.cumsumSeries.map (或Series.replace):

new_df = df.melt(var_name = 'status',value_name = 'time').sort_values('time')
new_df['counter'] = new_df['status'].map({'start':1,'end':-1}).cumsum()
print(new_df)
   status                time  counter
0   start 2018-01-01 00:47:00        1
1   start 2018-01-01 00:52:00        2
11    end 2018-01-01 00:54:00        1
2   start 2018-01-01 00:55:00        2
3   start 2018-01-01 00:57:00        3
13    end 2018-01-01 00:59:00        2
4   start 2018-01-01 01:00:00        3
12    end 2018-01-01 01:03:00        2
5   start 2018-01-01 01:07:00        3
15    end 2018-01-01 01:12:00        2
14    end 2018-01-01 01:16:00        1
16    end 2018-01-01 01:24:00        0
6   start 2018-01-01 01:33:00        1
7   start 2018-01-01 01:34:00        2
8   start 2018-01-01 01:37:00        3
9   start 2018-01-01 01:38:00        4
17    end 2018-01-01 01:38:00        3
10  start 2018-01-01 01:39:00        4
19    end 2018-01-01 01:41:00        3
20    end 2018-01-01 01:41:00        2
18    end 2018-01-01 01:47:00        1
21    end 2018-01-01 01:55:00        0
<小时/>

我们还可以使用numpy.cumsum :

new_df['counter'] = np.where(new_df['status'].eq('start'),1,-1).cumsum()

关于python - 如何计算 Pandas 中重叠的日期时间间隔?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60226735/

相关文章:

python - 硬币所有组合 - 两种算法之间的奇怪差异

python - 使用 shell 进行 Grepping - 长时间延迟

python - 如何将一个 Pandas 数据框列与日期结合,另一个与时间结合?

c# - 如何在 C# 中仅从 DateTime 序列化 Xml Date

sql - 对列值的更改 date_trunc 进行不同搜索

python - 使用 Python 使用 Google、Facebook 或 Twitter 帐户登录网站

python - 是否有自动安装 python 模块的方法(à la ruby​​gems)?

python - 将多列组合为单个日期时间的最佳方法是什么

pandas - 如何在使用 selenium 和 requests 的数千个下载程序中包含 try 和 Exceptions 测试?

python - 过滤异常值 - 如何使基于中值的 Hampel 函数更快?