在组内使用 pandas.shift()
Posted
技术标签:
【中文标题】在组内使用 pandas.shift()【英文标题】:Use pandas.shift() within a group 【发布时间】:2019-04-19 12:10:00 【问题描述】:我有一个包含面板数据的数据框,假设它是 100 个不同对象的时间序列:
object period value
1 1 24
1 2 67
...
1 1000 56
2 1 59
2 2 46
...
2 1000 64
3 1 54
...
100 1 451
100 2 153
...
100 1000 21
我想添加一个新列prev_value
,它将存储每个对象以前的value
:
object period value prev_value
1 1 24 nan
1 2 67 24
...
1 99 445 1243
1 1000 56 445
2 1 59 nan
2 2 46 59
...
2 1000 64 784
3 1 54 nan
...
100 1 451 nan
100 2 153 451
...
100 1000 21 1121
我可以使用 .shift() 和 .groupby() 以某种方式做到这一点吗?
【问题讨论】:
【参考方案1】:Pandas 的分组对象有一个groupby.DataFrameGroupBy.shift
方法,它将移动每个组中的指定列n periods
,就像常规数据框的shift
方法一样:
df['prev_value'] = df.groupby('object')['value'].shift()
对于以下示例数据框:
print(df)
object period value
0 1 1 24
1 1 2 67
2 1 4 89
3 2 4 5
4 2 23 23
结果是:
object period value prev_value
0 1 1 24 NaN
1 1 2 67 24.0
2 1 4 89 67.0
3 2 4 5 NaN
4 2 23 23 5.0
【讨论】:
请注意,预先对数据框进行排序更安全:df.sort_values(by=['period']).groupby('object')['value'].shift()
【参考方案2】:
如果您的 DataFrame 已经按分组键排序,您可以在整个 DataFrame 上使用单个 shift
和 where
到 NaN
溢出到下一组的行。对于具有许多组的较大 DataFrame,这可能会更快一些。
df['prev_value'] = df['value'].shift().where(df.object.eq(df.object.shift()))
object period value prev_value
0 1 1 24 NaN
1 1 2 67 24.0
2 1 4 89 67.0
3 2 4 5 NaN
4 2 23 23 5.0
一些与性能相关的时序:
import perfplot
import pandas as pd
import numpy as np
perfplot.show(
setup=lambda N: pd.DataFrame('object': np.repeat(range(N), 5),
'value': np.random.randint(1, 1000, 5*N)),
kernels=[
lambda df: df.groupby('object')['value'].shift(),
lambda df: df['value'].shift().where(df.object.eq(df.object.shift())),
],
labels=["GroupBy", "Where"],
n_range=[2 ** k for k in range(1, 22)],
equality_check=lambda x,y: np.allclose(x, y, equal_nan=True),
xlabel="# of Groups"
)
【讨论】:
以上是关于在组内使用 pandas.shift()的主要内容,如果未能解决你的问题,请参考以下文章
ROWNUMBER() OVER( PARTITION BY COL1 ORDER BY COL2)用法,先分组,然后在组内排名,分组计算等