我有一个包含2014年月度数据的数据框,用于317个股票代码系列(317个代码×12个月= DF中的3,804个行).我想将其转换为每日数据帧(317个代码x 365天= 115,705行).因此,我认为我需要在每个月的每一天传播月度值时进行上采样或重新索引,但我无法使其正常工作.
数据框目前采用以下格式:
>>> df
month ticker b c
2014-1 AAU 10 .04 #different values every month for each ticker
2014-2 AAU 20 .03
2014-3 AAU 13 .06
.
2014-12 AAU 11 .03
.
.
.
2014-1 ZZY 11 .11
2014-2 ZZY 6 .03
.
2014-12 ZZY 17 .09
Run Code Online (Sandbox Code Playgroud)
这就是我想要的:
>>> df
day ticker b c
2014-01-01 AAU 10 .04 #same values every day in month for each ticker
2014-01-02 AAU 10 .04
2014-01-03 AAU 10 .04
.
2014-01-31 AAU 10 .04
2014-02-01 AAU 20 .03
2014-02-02 AAU 20 .03
.
2014-02-28 AAU 20 .03
.
.
.
2014-12-30 ZZY 17 .09
2014-12-31 ZZY 17 .09
Run Code Online (Sandbox Code Playgroud)
我已经尝试过在白天结合重新采样进行组合,但更新后的数据框将从"2014-01-13"开始,而不是从1月1日开始,以"2014-12-01"而不是12月31日结束.我还尝试将月份值从"2014-1"更改为"2014-01-01"等,但重新采样的数据帧仍在"2014-01-01"结束.必须有一个更简单的方法,所以我很感激任何帮助.我整天都在圈子里四处走动.
unu*_*tbu 12
首先,将月份 - 日期字符串解析为Pandas时间戳:
df['month'] = pd.to_datetime(df['month'], format='%Y-%m')
# month ticker b c
# 0 2014-01-01 AAU 10 0.04
# 1 2014-02-01 AAU 20 0.03
# 2 2014-03-01 AAU 13 0.06
# 3 2014-12-01 AAU 11 0.03
# 4 2014-01-01 ZZY 11 0.11
# 5 2014-02-01 ZZY 6 0.03
# 6 2014-12-01 ZZY 17 0.09
Run Code Online (Sandbox Code Playgroud)
接下来,使用月份作为索引并将股票代码作为列级别来旋转DataFrame:
df = df.pivot(index='month', columns='ticker')
# b c
# ticker AAU ZZY AAU ZZY
# month
# 2014-01-01 10 11 0.04 0.11
# 2014-02-01 20 6 0.03 0.03
# 2014-03-01 13 NaN 0.06 NaN
# 2014-12-01 11 17 0.03 0.09
Run Code Online (Sandbox Code Playgroud)
通过现在的旋转,我们将能够在以后更容易地向前填充每个列.
现在找到开始和结束日期:
start_date = df.index.min() - pd.DateOffset(day=1)
end_date = df.index.max() + pd.DateOffset(day=31)
Run Code Online (Sandbox Code Playgroud)
有趣的是,请注意,添加pd.DateOffset(day=31)并不总是会导致日期在第31天结束.如果月份是2月,则添加将pd.DateOffset(day=31)返回2月的最后一天:
In [130]: pd.Timestamp('2014-2-28') + pd.DateOffset(day=31)
Out[130]: Timestamp('2014-02-28 00:00:00')
Run Code Online (Sandbox Code Playgroud)
这很好,因为这意味着添加pd.DateOffset(day=31)将始终为我们提供该月的最后一个有效日期.
现在我们可以重新索引并转发填充DataFrame:
dates = pd.date_range(start_date, end_date, freq='D')
dates.name = 'date'
df = df.reindex(dates, method='ffill')
Run Code Online (Sandbox Code Playgroud)
产量
In [160]: df.head()
Out[160]:
b c
ticker AAU ZZY AAU ZZY
date
2014-01-01 10 11 0.04 0.11
2014-01-02 10 11 0.04 0.11
2014-01-03 10 11 0.04 0.11
2014-01-04 10 11 0.04 0.11
2014-01-05 10 11 0.04 0.11
In [161]: df.tail()
Out[161]:
b c
ticker AAU ZZY AAU ZZY
date
2014-12-27 11 17 0.03 0.09
2014-12-28 11 17 0.03 0.09
2014-12-29 11 17 0.03 0.09
2014-12-30 11 17 0.03 0.09
2014-12-31 11 17 0.03 0.09
Run Code Online (Sandbox Code Playgroud)
要将滚动条移出列索引并返回到列中:
df = df.stack('ticker')
df = df.sortlevel(level=1)
df = df.reset_index()
Run Code Online (Sandbox Code Playgroud)
所以把它们放在一起:
import pandas as pd
df = pd.read_table('data', sep='\s+')
df['month'] = pd.to_datetime(df['month'], format='%Y-%m')
df = df.pivot(index='month', columns='ticker')
start_date = df.index.min() - pd.DateOffset(day=1)
end_date = df.index.max() + pd.DateOffset(day=31)
dates = pd.date_range(start_date, end_date, freq='D')
dates.name = 'date'
df = df.reindex(dates, method='ffill')
df = df.stack('ticker')
df = df.sortlevel(level=1)
df = df.reset_index()
Run Code Online (Sandbox Code Playgroud)
产量
In [163]: df.head()
Out[163]:
date ticker b c
0 2014-01-01 AAU 10 0.04
1 2014-01-02 AAU 10 0.04
2 2014-01-03 AAU 10 0.04
3 2014-01-04 AAU 10 0.04
4 2014-01-05 AAU 10 0.04
In [164]: df.tail()
Out[164]:
date ticker b c
450 2014-12-27 ZZY 17 0.09
451 2014-12-28 ZZY 17 0.09
452 2014-12-29 ZZY 17 0.09
453 2014-12-30 ZZY 17 0.09
454 2014-12-31 ZZY 17 0.09
Run Code Online (Sandbox Code Playgroud)
小智 5
我们来做一个综合实验。假设我们有每日时间序列数据:
dates = pd.date_range(start, end, freq='D')
ts = pd.Series(data, index=dates)
Run Code Online (Sandbox Code Playgroud)
通过对一个月内的所有数据进行平均来生成每月时间序列:
ts_mon = ts.resample('MS', how='mean')
Run Code Online (Sandbox Code Playgroud)
现在尝试将此每月时间序列上采样回每日时间序列,并在一个月内使用统一值。第一种使用 reindex 从 @unutbu 借用步骤的方法效果很好:
ts_daily = ts_mon.reindex(dates, method='ffill')
Out:
2000-01-01 100.21
2000-01-02 100.21
...
2000-12-30 80.75
2000-12-31 80.75
Run Code Online (Sandbox Code Playgroud)
使用 resample 的第二种方法不起作用,因为它返回上个月的第一天:
ts_daily = ts_mon.resample('D').ffill()
Out:
2000-01-01 100.21
2000-01-02 100.21
...
2000-11-30 99.33
2000-12-01 80.75
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
6015 次 |
| 最近记录: |