对《利用Python 进行数据分析》(Wes Mckinney著)一书中的第十章中时间序列进行代码实验。原书中采用的是Python2.7,而我采用的Python3.7在Pycharm调试的,因此对书中源代码进行了一定的修改,每步结果与原文校验对照一致(除了随机函数外;输出结果在注释中,简单的输出就没写结果),全手工敲写,供参考。
Pdf文档和数据集参见:《利用Python 进行数据分析》第二章:引言中的分析代码(含pdf和数据集下载链接)
# -*- coding:utf-8 -*-
from datetime import datetime, timedelta
import pandas as pd
import numpy as np
from pandas import DataFrame, Series
时间序列数据的意义取决于具体的应用场景
时间戳(timestamp):特定的时刻
固定时期(period):如2007年1月或2010年全年
时间间隔(interval):由起始和结束时间戳表示,时期(period)可以被看作间隔的特例
# 主要用到datetime、time以及calendar模块
now = datetime.now()
print(now) # 2020-09-28 14:05:42.871960
print(now.year, now.month, now.day) # 2020 9 28
# datetime以毫秒形式储存日期和时间
delta = datetime(2011,1,7) - datetime(2008, 6, 24, 8, 15)
print(delta) # 926 days, 15:45:00
# datetime.timedelta表示两个datetime对象之间的时间差
timedelta(926, 56700)
print(delta.days) # 926
print(delta.seconds) # 56700
# 可以给datetime对象加上(减去)一个或多个timedelta,会产生一个新对象
start = datetime(2011,1,7)
ret = start + timedelta(12)
print(ret) # 2011-01-19 00:00:00
ret = start - 2 * timedelta(12)
print(ret) # 2010-12-14 00:00:00
# 利用str或strftime方法(传入格式化字符串),datetime对象和pandas的Timestamp对象可以被格式化为字符串
stamp = datetime(2011, 1, 3)
print(str(stamp)) # 2011-01-03 00:00:00
print(stamp.strftime('%Y-%m-%d')) # 2011-01-03
# date.time.strptime也可以用这些格式化编码将字符串转化为日期
value = '2011-01-09'
print(datetime.strptime(value,'%Y-%m-%d')) # 2011-01-09 00:00:00
datestrs=['7/6/2011', '8/6/2011']
print([datetime.strptime(x, '%m/%d/%Y') for x in datestrs])
'''[datetime.datetime(2011, 7, 6, 0, 0), datetime.datetime(2011, 8, 6, 0, 0)]'''
# datetime.strptime是通过已知格式进行日期解析,但每次编写都需要定义格式比较麻烦
# 所以我们可以使用dateutil 这个第三方库的parser.parse方法
from dateutil.parser import parse
print(parse('2011-01-03')) # 2011-01-03 00:00:00
# dateutil 可以解析几乎所有人类能理解的日期表现形式
print(parse('Jan 31, 1997 10:45 PM')) # 1997-01-31 22:45:00
# 国际通用格式中,日常常出现在月的前面,传入dayfirst=True即可解决这个问题
print(parse('6/12/2011', dayfirst=True)) # 2011-12-06 00:00:00
# to_datetime方法可以解析多种不同日期的表示形式
print(datestrs) # ['7/6/2011', '8/6/2011']
print(pd.to_datetime(datestrs))
'''
DatetimeIndex(['2011-07-06', '2011-08-06'], dtype='datetime64[ns]', freq=None)
'''
# to_datetime也可以处理缺失值(None、空字符串等)
idx = pd.to_datetime(datestrs + [None])
print(idx)
'''
atetimeIndex(['2011-07-06', '2011-08-06', 'NaT'], dtype='datetime64[ns]', freq=None)
'''
print(idx[2]) # NaT , NaT(Not a Time)是pandas中时间戳数据的NA值
print(pd.isnull(idx)) # [False False True]
dates = [datetime(2011,1,2), datetime(2011, 1, 5), datetime(2011,1,7),
datetime(2011,1,8), datetime(2011,1,10), datetime(2011,1,12)]
ts = Series(np.random.randn(6), index = dates)
print(ts)
'''
2011-01-02 -0.804594
2011-01-05 0.444492
2011-01-07 -1.336713
2011-01-08 1.380549
2011-01-10 -1.090957
2011-01-12 0.162639
dtype: float64
'''
# datetime对象是被放在一个DatetimeIndex中,现在ts就成为一个TimeSeries了
print(type(ts))
'''
DatetimeIndex(['2011-01-02', '2011-01-05', '2011-01-07', '2011-01-08',
'2011-01-10', '2011-01-12'],
dtype='datetime64[ns]', freq=None)
'''
print(ts.index)
'''
2011-01-02 -1.609187
2011-01-05 NaN
2011-01-07 -2.673426
2011-01-08 NaN
2011-01-10 -2.181915
2011-01-12 NaN
dtype: float64
'''
# 跟其他Series一样,不同索引的时间序列之间的算术运算会按自动日期对齐
print(ts + ts[::2])
print(ts.index.dtype) # datetime64[ns]
# DatetimeIndex中的各个标量值是pandas的Timestamp对象
stamp = ts.index[0]
print(stamp) # 2011-01-02 00:00:00
dates = [datetime(2011,1,2), datetime(2011, 1, 5), datetime(2011,1,7),
datetime(2011,1,8), datetime(2011,1,10), datetime(2011,1,12)]
ts = Series(np.random.randn(6), index = dates)
print(ts)
'''
2011-01-02 0.348253
2011-01-05 -0.068450
2011-01-07 -1.073036
2011-01-08 1.059299
2011-01-10 0.497196
2011-01-12 0.713568
dtype: float64
'''
# TimeSeries是Series的一个类,所以在索引以及数据选取方面他们的行为是一样的
stamp=ts.index[2]
print(ts[stamp]) # -1.073035582647907
# 可以传入一个可以解释为日期的字符串
print(ts['1/10/2011']) # 0.4971955016152246
print(ts['20110110']) # 0.4971955016152246
# 对于较长的时间序列,只需要传入“年”或“年月”即可轻松选取数据的切片
longer_ts = Series(np.random.randn(1000),
index=pd.date_range('1/1/2000', periods=1000))
print(longer_ts)
'''
2000-01-01 0.650802
2000-01-02 2.018351
2000-01-03 0.676741
2000-01-04 0.779642
2000-01-05 0.851207
...
2002-09-22 -1.794156
2002-09-23 0.515699
2002-09-24 0.257113
2002-09-25 -1.512441
2002-09-26 -0.680429
Freq: D, Length: 1000, dtype: float64
'''
print(longer_ts['2001'])
'''
2001-01-01 1.357131
2001-01-02 -0.840957
2001-01-03 -1.000980
2001-01-04 -1.183331
2001-01-05 0.453523
...
2001-12-27 0.919488
2001-12-28 -1.240291
2001-12-29 0.061306
2001-12-30 -1.226537
2001-12-31 -0.744249
Freq: D, Length: 365, dtype: float64
'''
print(longer_ts['2001-05'])
'''
2001-05-01 0.377640
2001-05-02 0.389160
2001-05-03 -0.657888
2001-05-04 1.353799
2001-05-05 0.834874
...
2001-05-26 -1.333958
2001-05-27 1.405335
2001-05-28 -0.217538
2001-05-29 -0.029023
2001-05-30 -0.889619
2001-05-31 -0.986640
Freq: D, dtype: float64
'''
# 通过日期进行切片的方式只对规则Series有效
print(ts[datetime(2011,1,7):])
'''
2011-01-07 -1.073036
2011-01-08 1.059299
2011-01-10 0.497196
2011-01-12 0.713568
dtype: float64
'''
# 由于大部分时间都是按照时间先后排序,因此可以用不存在于该时间序列中的时间戳对其进行切片
print(ts)
'''
2011-01-02 0.348253
2011-01-05 -0.068450
2011-01-07 -1.073036
2011-01-08 1.059299
2011-01-10 0.497196
2011-01-12 0.713568
dtype: float64
'''
print(ts['1/6/2011':'1/11/2011']) # 取范围内的日期
'''
2011-01-07 -1.073036
2011-01-08 1.059299
2011-01-10 0.497196
dtype: float64
'''
# 截取两个日期之间的TimeSeries
print(ts.truncate(after='1/9/2011'))
'''
2011-01-02 0.348253
2011-01-05 -0.068450
2011-01-07 -1.073036
2011-01-08 1.059299
dtype: float64
'''
# 也可以对DataFrame操作,对DataFrame的行进行索引
dates = pd.date_range('1/1/2000',periods = 100, freq='W-WED')
long_df = DataFrame(np.random.randn(100,4),
index=dates,
columns=['Colorado','Texas', 'NewYork', 'Ohio'])
print(long_df.loc['5-2001'])
'''
Colorado Texas NewYork Ohio
2001-05-02 1.467936 1.063116 1.344797 -0.580989
2001-05-09 0.637778 -0.905873 0.855643 -1.161038
2001-05-16 0.305796 -1.233853 -0.628636 -0.052159
2001-05-23 -1.098029 0.052049 0.531545 1.161001
2001-05-30 -0.981410 -2.068461 2.049203 -0.786793
'''
dates = pd.DatetimeIndex(['1/1/2000', '1/2/2000', '1/2/2000','1/2/2000', '1/3/2000'])
dup_ts = Series(np.arange(5), index=dates)
print(dup_ts)
'''
2000-01-01 0
2000-01-02 1
2000-01-02 2
2000-01-02 3
2000-01-03 4
dtype: int32
'''
# 通过检查索引的is_unique属性,可以知道它是不是唯一的
print(dup_ts.index.is_unique) # False
# 通过对这个时间序列进行索引,要么产生标量值,要么产生切片,取决于所选的时间点是否重复
print(dup_ts['1/3/2000']) # 4
print(dup_ts['1/2/2000'])
'''
2000-01-02 1
2000-01-02 2
2000-01-02 3
dtype: int32
'''
# 如果想要对具有非唯一的数据进行聚合,可以使用groupby,并传入level=0(索引的唯一一层!)
grouped = dup_ts.groupby(level=0)
print(grouped.mean())
'''
2000-01-01 0
2000-01-02 2
2000-01-03 4
dtype: int32
'''
print(grouped.count())
'''
2000-01-01 1
2000-01-02 3
2000-01-03 1
dtype: int64
'''
# pandas有一套标准时间序列频率以及重采样、频率推断、生成固定频率日期的范围
dates = [datetime(2011,1,2), datetime(2011, 1, 5), datetime(2011,1,7),
datetime(2011,1,8), datetime(2011,1,10), datetime(2011,1,12)]
ts = Series(np.random.randn(6), index = dates)
print(ts)
ts_resmp = ts.resample('D')
print(ts_resmp)
'''DatetimeIndexResampler [freq=, axis=0, closed=left, label=left, convention=start, base=0]'''
ts_resmp_sum = ts.resample('3D').sum()
print(ts_resmp_sum) # 按3天重新采样并求和
'''
2011-01-02 1.041772
2011-01-05 -0.854215
2011-01-08 -2.727751
2011-01-11 0.809483
Freq: 3D, dtype: float64
'''
# 关于重采样是比较大的主题,在第6小节专门讨论
# 用pandas_range可用于生成指定长度的DatetimeIndex
index = pd.date_range('4/1/2012', '6/1/2012')
print(index[:5])
'''
DatetimeIndex(['2012-04-01', '2012-04-02', '2012-04-03', '2012-04-04',
'2012-04-05'],
dtype='datetime64[ns]', freq='D')
'''
# 默认情况下,date_range会按天计算的时间点
# 如果传入起始或起始结束日期,还需要传入一个表示一段时间的数字
print(pd.date_range(start='4/1/2012', periods=20))
'''
DatetimeIndex(['2012-04-01', '2012-04-02', '2012-04-03', '2012-04-04',
'2012-04-05', '2012-04-06', '2012-04-07', '2012-04-08',
'2012-04-09', '2012-04-10', '2012-04-11', '2012-04-12',
'2012-04-13', '2012-04-14', '2012-04-15', '2012-04-16',
'2012-04-17', '2012-04-18', '2012-04-19', '2012-04-20'],
dtype='datetime64[ns]', freq='D')
'''
print(pd.date_range(end='6/1/2012',periods =20))
'''
DatetimeIndex(['2012-05-13', '2012-05-14', '2012-05-15', '2012-05-16',
'2012-05-17', '2012-05-18', '2012-05-19', '2012-05-20',
'2012-05-21', '2012-05-22', '2012-05-23', '2012-05-24',
'2012-05-25', '2012-05-26', '2012-05-27', '2012-05-28',
'2012-05-29', '2012-05-30', '2012-05-31', '2012-06-01'],
dtype='datetime64[ns]', freq='D')
'''
# 生成一个由每月最后一个工作日组成的日期索引,传入“BM"频率(business end of month)
print(pd.date_range('1/1/2000','12/1/2000',freq='BM'))
'''
DatetimeIndex(['2000-01-31', '2000-02-29', '2000-03-31', '2000-04-28',
'2000-05-31', '2000-06-30', '2000-07-31', '2000-08-31',
'2000-09-29', '2000-10-31', '2000-11-30'],
dtype='datetime64[ns]', freq='BM')
'''
# date_range默认保留起始和结束时间戳的时间信息(如果有的话)
print(pd.date_range('5/2/2012 12:56:31', periods=5))
'''
DatetimeIndex(['2012-05-02 12:56:31', '2012-05-03 12:56:31',
'2012-05-04 12:56:31', '2012-05-05 12:56:31',
'2012-05-06 12:56:31'],
dtype='datetime64[ns]', freq='D')
'''
# normalize选项可以实现产生一组被规范化到午夜的时间戳
print(pd.date_range('5/2/2012 12:56:31', periods=5, normalize=True))
'''
DatetimeIndex(['2012-05-02', '2012-05-03', '2012-05-04', '2012-05-05',
'2012-05-06'],
dtype='datetime64[ns]', freq='D')
'''
# pandas中的频率是由一个基础频率和一个乘数组成的
# 基础频率通常以一个字符串别名表示,比如“M"表示每月,”H“表示每小时
from pandas.tseries.offsets import Hour, Minute
hour = Hour()
print(hour) #
# 传入一个整数即可定义便宜量的倍数
four_hours = Hour(4)
print(four_hours) # <4 * Hours>
# 在基础频率前面放上一个整数即可创建倍数
print(pd.date_range('1/1/2000', '1/1/2000 23:59', freq='4h'))
'''
DatetimeIndex(['2000-01-01 00:00:00', '2000-01-01 04:00:00',
'2000-01-01 08:00:00', '2000-01-01 12:00:00',
'2000-01-01 16:00:00', '2000-01-01 20:00:00'],
dtype='datetime64[ns]', freq='4H')
'''
# 大部分偏移量对象都可以通过加法进行连接
print(Hour(2) + Minute(30)) # <150 * Minutes>
# 同时也可以传入频率字符串(如“2h30min”),这种字符串可以被高效地解析为等效的表达式
print(pd.date_range('1/1/2000',periods = 10, freq='1h30min'))
'''
DatetimeIndex(['2000-01-01 00:00:00', '2000-01-01 01:30:00',
'2000-01-01 03:00:00', '2000-01-01 04:30:00',
'2000-01-01 06:00:00', '2000-01-01 07:30:00',
'2000-01-01 09:00:00', '2000-01-01 10:30:00',
'2000-01-01 12:00:00', '2000-01-01 13:30:00'],
dtype='datetime64[ns]', freq='90T')
'''
# WOM日期,week of month是一种非常实用的频率类,如获得诸如“每月第3个星期五”之类的日期
rng = pd.date_range('1/1/2012','9/1/2012', freq="WOM-3FRI")
print(rng)
'''
DatetimeIndex(['2012-01-20', '2012-02-17', '2012-03-16', '2012-04-20',
'2012-05-18', '2012-06-15', '2012-07-20', '2012-08-17'],
dtype='datetime64[ns]', freq='WOM-3FRI')
'''
# 移动是指沿着时间轴将数据前移或后移,Series和DataFrame都有一个shift方法用于执行单纯的前移或后移操作
ts = Series(np.random.randn(4),
index = pd.date_range('1/1/2000', periods=4, freq='M'))
print(ts)
'''
2000-01-31 -0.309081
2000-02-29 0.754501
2000-03-31 -0.727029
2000-04-30 -0.628417
Freq: M, dtype: float64
'''
# Shift通常用于计算一个时间序列或多个时间序列中百分比变化:ts/st.shift(1) - 1
# 如果频率已知,则可以将其传给shift以便实现对时间戳进行位移而不是对数据进行简单位移
print(ts.shift(2, freq='M'))
'''
2000-03-31 -0.309081
2000-04-30 0.754501
2000-05-31 -0.727029
2000-06-30 -0.628417
Freq: M, dtype: float64
'''
# 还可以使用其他频率,可以灵活对数据进行超前或滞后处理
print(ts.shift(3,freq='D'))
'''
2000-02-03 -0.309081
2000-03-03 0.754501
2000-04-03 -0.727029
2000-05-03 -0.628417
dtype: float64
'''
print(ts.shift(1,freq='3D'))
'''
2000-02-03 -0.309081
2000-03-03 0.754501
2000-04-03 -0.727029
2000-05-03 -0.628417
dtype: float64
'''
print(ts.shift(1,freq='90T')) # 1h30mins
'''
2000-01-31 01:30:00 -0.309081
2000-02-29 01:30:00 0.754501
2000-03-31 01:30:00 -0.727029
2000-04-30 01:30:00 -0.628417
Freq: M, dtype: float64
'''
print('----')
# 通过偏移量对日期进行位移
from pandas.tseries.offsets import Day, MonthEnd
now = datetime(2011, 11, 17)
print(now + 3*Day()) # 2011-11-20 00:00:00
# 如果加的是锚点偏移量(MonthEnd例如),第一次增量会将原日期向前滚动到符合频率规则的下一日期
print(now+MonthEnd()) # 2011-11-30 00:00:00
print(now+MonthEnd(2)) # 2011-12-31 00:00:00
# 通过锚点偏移量的rollforward和rollback方法,可显式地将日期向前或向后”滚动“
offset = MonthEnd()
print(offset.rollforward(now)) # 2011-11-30 00:00:00
print(offset.rollback(now)) # 2011-10-31 00:00:00
# 日期偏移量还有一个巧妙的用法,即结合groupby使用这两个“滚动”方法
ts = Series(np.random.randn(20),
index = pd.date_range('1/15/2000',periods=20,freq='4d'))
print(ts.groupby(offset.rollforward).mean())
'''
2000-01-31 0.168758
2000-02-29 -0.167549
2000-03-31 0.379540
dtype: float64
'''
# 当然实现上述功能最快的方法是使用resample函数
print(ts.resample("M").mean())
'''
2000-01-31 0.168758
2000-02-29 -0.167549
2000-03-31 0.379540
Freq: M, dtype: float64
'''
# 时区信息来自第三方库Pytz
import pytz
print(pytz.common_timezones[-5:]) # ['US/Eastern', 'US/Hawaii', 'US/Mountain', 'US/Pacific', 'UTC']
# 从pytz中获取时区对象,使用pytz.timezone即可
tz=pytz.timezone('US/Eastern')
print(tz) # US/Eastern
# 默认情况,pandas的时间序列是单纯的(naive)时区
rng = pd.date_range('3/9/2012 9:30', periods=6, freq='D')
ts=Series(np.random.randn(len(rng)), index = rng)
print(ts.index.tz) # None
# 在生成日期范围的时候还可以加上一个时区集
print(pd.date_range('2/9/2012 9:30', periods=10, freq='D', tz='UTC'))
'''
DatetimeIndex(['2012-02-09 09:30:00+00:00', '2012-02-10 09:30:00+00:00',
'2012-02-11 09:30:00+00:00', '2012-02-12 09:30:00+00:00',
'2012-02-13 09:30:00+00:00', '2012-02-14 09:30:00+00:00',
'2012-02-15 09:30:00+00:00', '2012-02-16 09:30:00+00:00',
'2012-02-17 09:30:00+00:00', '2012-02-18 09:30:00+00:00'],
dtype='datetime64[ns, UTC]', freq='D')
'''
# 从单纯到本地化的转换时通过tz_local方法处理的
ts_utc=ts.tz_localize('UTC')
print(ts_utc)
'''
2012-03-09 09:30:00+00:00 -0.366451
2012-03-10 09:30:00+00:00 -1.254051
2012-03-11 09:30:00+00:00 0.733324
2012-03-12 09:30:00+00:00 -0.267528
2012-03-13 09:30:00+00:00 -0.938285
2012-03-14 09:30:00+00:00 -1.037081
Freq: D, dtype: float64
'''
print(ts_utc.index)
'''
DatetimeIndex(['2012-03-09 09:30:00+00:00', '2012-03-10 09:30:00+00:00',
'2012-03-11 09:30:00+00:00', '2012-03-12 09:30:00+00:00',
'2012-03-13 09:30:00+00:00', '2012-03-14 09:30:00+00:00'],
dtype='datetime64[ns, UTC]', freq='D')
'''
# 一旦时间序列被本地化到某个特定时区,就可以用tz_convert将其转换到别的时区
print(ts_utc.tz_convert('US/Eastern'))
'''
2012-03-09 04:30:00-05:00 -0.366451
2012-03-10 04:30:00-05:00 -1.254051
2012-03-11 05:30:00-04:00 0.733324
2012-03-12 05:30:00-04:00 -0.267528
2012-03-13 05:30:00-04:00 -0.938285
2012-03-14 05:30:00-04:00 -1.037081
Freq: D, dtype: float64
'''
# 对于上面的时间序列(跨越了美国东部时区的夏令时期转变期)可以先将其本地化到EST,再转为UTC或柏林时间
ts_eastern = ts.tz_localize('US/Eastern')
print(ts_eastern.tz_convert('UTC'))
'''
2012-03-09 14:30:00+00:00 -0.366451
2012-03-10 14:30:00+00:00 -1.254051
2012-03-11 13:30:00+00:00 0.733324
2012-03-12 13:30:00+00:00 -0.267528
2012-03-13 13:30:00+00:00 -0.938285
2012-03-14 13:30:00+00:00 -1.037081
dtype: float64
'''
print(ts_eastern.tz_convert('Europe/Berlin'))
'''
2012-03-09 15:30:00+01:00 -0.366451
2012-03-10 15:30:00+01:00 -1.254051
2012-03-11 14:30:00+01:00 0.733324
2012-03-12 14:30:00+01:00 -0.267528
2012-03-13 14:30:00+01:00 -0.938285
2012-03-14 14:30:00+01:00 -1.037081
dtype: float64
'''
# tz_localize和tz_convert也是DatetimeIndex的实例方法
print(ts.index.tz_localize('Asia/Shanghai'))
'''
DatetimeIndex(['2012-03-09 09:30:00+08:00', '2012-03-10 09:30:00+08:00',
'2012-03-11 09:30:00+08:00', '2012-03-12 09:30:00+08:00',
'2012-03-13 09:30:00+08:00', '2012-03-14 09:30:00+08:00'],
dtype='datetime64[ns, Asia/Shanghai]', freq=None)
'''
# 与时间序列和日期范围差不多,Timestamp对象也能从单纯型(naive)本地化为时区意识型,并从一个时区转换到另一时区
stamp = pd.Timestamp('2011-03-12 04:00')
stamp_utc = stamp.tz_localize('utc')
print(stamp_utc.tz_convert('US/Eastern')) # 2011-03-11 23:00:00-05:00
# 在创建Timestamp时,还可以传入一个时区信息
stamp_moscow=pd.Timestamp('2011-03-12 04:00', tz='Europe/Moscow')
print(stamp_moscow) # 2011-03-12 04:00:00+03:00
# 时区意识型Timestamp对象在内部保存了一个UTC时间戳值,这个值在时区转换过程中是不会发生变化的
print(stamp_utc.value) # 1299902400000000000
print(stamp_utc.tz_convert('US/Eastern').value) # 1299902400000000000
# 使用pandas的DateOffset对象执行时间算术运算时,运算过程会自动关注是否存在夏令时转变期
# 夏令时转变前30分钟
from pandas.tseries.offsets import Hour
stamp = pd.Timestamp('2012-03-12 01:30', tz='US/Eastern')
print(stamp) # 2012-03-12 01:30:00-04:00
print(stamp+Hour()) # 2012-03-12 02:30:00-04:00
# 夏令时转变前90分钟
stamp=pd.Timestamp('2012-11-04 00:30', tz='US/Eastern')
print(stamp) # 2012-11-04 00:30:00-04:00
print(stamp + 2*Hour()) # 2012-11-04 01:30:00-05:00
# 如果两个时间序列的时区不同,将它们合并到一起时,最终结果就会是UTC
# 由于时间戳其实是以UTC储存的,所以这是一个简单的运算,并不需要发生任何转换
rng = pd.date_range('3/7/2012 09:30', periods = 10, freq='B')
ts = Series(np.random.randn(len(rng)), index=rng)
print(ts)
'''
2012-03-07 09:30:00 0.041705
2012-03-08 09:30:00 0.461161
2012-03-09 09:30:00 0.197227
2012-03-12 09:30:00 -1.409566
2012-03-13 09:30:00 0.227489
2012-03-14 09:30:00 -1.624908
2012-03-15 09:30:00 0.717115
2012-03-16 09:30:00 -1.355306
2012-03-19 09:30:00 -1.684638
2012-03-20 09:30:00 -0.566004
Freq: B, dtype: float64
'''
ts1= ts[:7].tz_localize('Europe/London')
ts2= ts1[2:].tz_convert('Europe/Moscow')
result = ts1 + ts2
print(result.index)
'''
DatetimeIndex(['2012-03-07 09:30:00+00:00', '2012-03-08 09:30:00+00:00',
'2012-03-09 09:30:00+00:00', '2012-03-12 09:30:00+00:00',
'2012-03-13 09:30:00+00:00', '2012-03-14 09:30:00+00:00',
'2012-03-15 09:30:00+00:00'],
dtype='datetime64[ns, UTC]', freq=None)
'''
# 时期表示的是时间区间,如日、数月、数季、数年等
# Period类所表示的就是此种类型,其构造函数需要用到一个字符串或整数,以及频率
p = pd.Period(2007,freq='A-DEC')
print(p) # 2007
# 上述p值表示的是2007年1月1日到2007年12月31日之间的整段时间
# 对Period对象加上或减去一个整数即可达到根据其频率进行位移的效果
print(p+5) # 2012
print(p-2) # 2005
# 如果两个Period对象拥有相同的频率,则他们的差就是他们之间的单位数量
print(pd.Period('2014', freq='A-DEC') - p) # <7 * YearEnds: month=12>
# period_range函数可以用于创建规则的的时期范围
rng = pd.period_range('1/1/2000','6/30/2000', freq='M')
print(rng)
'''
PeriodIndex(['2000-01', '2000-02', '2000-03', '2000-04', '2000-05', '2000-06'], dtype='period[M]', freq='M')
'''
# PeriodIndex类保存了一组Period,可以在任何pandas数据结构中被用作轴索引
print(Series(np.random.randn(6), index = rng))
'''
2000-01 -2.155357
2000-02 -0.912094
2000-03 0.358419
2000-04 0.337311
2000-05 1.036003
2000-06 -0.613236
Freq: M, dtype: float64
'''
# PeriodIndex类的构造函数还允许直接使用一组字符串
values = ['2001Q3', '2002Q2', '2003Q1']
index = pd.PeriodIndex(values, freq='Q-DEC')
print(index)
'''
PeriodIndex(['2001Q3', '2002Q2', '2003Q1'], dtype='period[Q-DEC]', freq='Q-DEC')
'''
# Period和PeriodIndex都可以通过asfreq方法被转换成别的频率
p = pd.Period('2007', freq='A-DEC')
print(p) # 2007
# 转换为一个年初或年末的一个月度时期
print(p.asfreq('M',how='start')) # 2007-01
print(p.asfreq('M',how='end')) # 2007-12
# Period('2007','A-DEC')可以看做一个被划分为多个月度时期的时间段中的游标
p=pd.Period('2007', freq='A-JUN')
print(p.asfreq('M', 'start')) # 2006-07
print(p.asfreq('M', 'end')) # 2007-06
# 高频率转换为低频率时,超时期是由时期所属的位置决定的
# 如,在A-JUN频率中,月份“2007年8月”实际上是属于周期“2008年”的
p=pd.Period('2007-08','M')
print(p.asfreq('A-JUN')) # 2008
# PeriodIndex或TimeSeries的评率转换方式也是如此
rng =pd.period_range('2006','2009', freq='A-DEC')
ts=Series(np.random.randn(len(rng)), index=rng)
print(ts)
'''
2006 -0.718306
2007 0.273010
2008 -0.441507
2009 -1.229443
Freq: A-DEC, dtype: float64
'''
print(ts.asfreq('M', how='start'))
'''
2006-01 -0.718306
2007-01 0.273010
2008-01 -0.441507
2009-01 -1.229443
Freq: M, dtype: float64
'''
print(ts.asfreq('M', how='end'))
'''
2006-12 -0.718306
2007-12 0.273010
2008-12 -0.441507
2009-12 -1.229443
Freq: M, dtype: float64
'''
# pandas支持12中可能的季度型频率,即Q-JAN到Q-DEC
# 我的理解,频率是哪个月份,就是那年Q4结束的月份,如本例
p=pd.Period('2012Q4', freq='Q-JAN')
print(p) # 2012Q4
# 在以1月结束的财年中,2012Q4是从11月到1月
print(p.asfreq('D', 'start')) # 2011-11-01
print(p.asfreq('D', 'end')) # 2012-01-31
# 取该季度倒数第二个工作日下午4点的时间戳
p4pm =(p.asfreq('B', 'e') - 1).asfreq('T', 's') + 16*60
print(p4pm) # 2012-01-30 16:00
print(p4pm.to_timestamp()) # 2012-01-30 16:00:00
# period_range可以用于生产季度型范围
rng = pd.period_range('2011Q3','2012Q4', freq='Q-JAN')
ts=Series(np.arange(len(rng)), index = rng)
print(ts)
'''
2011Q3 0
2011Q4 1
2012Q1 2
2012Q2 3
2012Q3 4
2012Q4 5
Freq: Q-JAN, dtype: int32
'''
new_rng=(rng.asfreq('B', 'e') -1).asfreq('T', 's') + 16*60
ts.index=new_rng.to_timestamp()
print(ts)
'''
2010-10-28 16:00:00 0
2011-01-28 16:00:00 1
2011-04-28 16:00:00 2
2011-07-28 16:00:00 3
2011-10-28 16:00:00 4
2012-01-30 16:00:00 5
dtype: int32
'''
# 通过使用to_period方法,可以将时间戳索引的Series和DataFrame对象转为以时期为索引
rng = pd.date_range('1/1/2000',periods=3,freq='M')
ts = Series(np.random.randn(3), index=rng)
pts=ts.to_period()
print(ts)
'''
2000-01-31 0.563108
2000-02-29 0.784912
2000-03-31 1.014484
Freq: M, dtype: float64
'''
print(pts)
'''
2000-01 0.563108
2000-02 0.784912
2000-03 1.014484
Freq: M, dtype: float64
'''
# 由于时期指的是非重叠时间区间,因此对于给定的频率,一个时间戳只能属于一个时期
rng = pd.date_range('1/29/2000', periods=6, freq='D')
ts2=Series(np.random.randn(6), index=rng)
print(ts2.to_period('M'))
'''
2000-01 -0.139087
2000-01 -0.136360
2000-01 -2.787923
2000-02 -1.520740
2000-02 -0.473269
2000-02 0.600253
Freq: M, dtype: float64
'''
pts=ts.to_period()
print(pts)
'''
2000-01 -0.219983
2000-02 -1.073624
2000-03 -0.681099
Freq: M, dtype: float64
'''
# 转化为时间戳,使用to_timestamp
print(pts.to_timestamp(how='end'))
'''
2000-01-31 23:59:59.999999999 -0.219983
2000-02-29 23:59:59.999999999 -1.073624
2000-03-31 23:59:59.999999999 -0.681099
dtype: float64
'''
# 固定频率的数据集通常会将时间信息分开存放在多列中
data = pd.read_csv('python_data/ch08/macrodata.csv')
print(data.year)
'''
0 1959.0
1 1959.0
2 1959.0
3 1959.0
4 1960.0
...
198 2008.0
199 2008.0
200 2009.0
201 2009.0
202 2009.0
Name: year, Length: 203, dtype: float64
'''
print(data.quarter)
'''
0 1.0
1 2.0
2 3.0
3 4.0
4 1.0
...
198 3.0
199 4.0
200 1.0
201 2.0
202 3.0
Name: quarter, Length: 203, dtype: float64
'''
# 将两个数组以及一个频率传入PeriodIndex,可以将它们合并成DataFrame的一个索引
index = pd.PeriodIndex(year=data.year, quarter=data.quarter, freq='Q-DEC')
print(index)
'''
PeriodIndex(['1959Q1', '1959Q2', '1959Q3', '1959Q4', '1960Q1', '1960Q2',
'1960Q3', '1960Q4', '1961Q1', '1961Q2',
...
'2007Q2', '2007Q3', '2007Q4', '2008Q1', '2008Q2', '2008Q3',
'2008Q4', '2009Q1', '2009Q2', '2009Q3'],
dtype='period[Q-DEC]', length=203, freq='Q-DEC')
'''
data.index = index
print(data.infl) # infl为data的其中一个属性
'''
1959Q1 0.00
1959Q2 2.34
1959Q3 2.74
1959Q4 0.27
1960Q1 2.31
...
2008Q3 -3.16
2008Q4 -8.79
2009Q1 0.94
2009Q2 3.37
2009Q3 3.56
Freq: Q-DEC, Name: infl, Length: 203, dtype: float64
'''
# 重采样是将时间序列从一个频率转换到另一个评率的过程
# 将高频率数据聚合到低频率称为讲采样,而将低频率数据转换到高频率则称为升采样
rng = pd.date_range('1/1/2000',periods = 100, freq='D')
ts = Series(np.random.randn(len(rng)), index=rng)
print(ts.resample('M').mean())
'''
2000-01-31 0.066587
2000-02-29 -0.240131
2000-03-31 -0.126769
2000-04-30 0.387274
Freq: M, dtype: float64
'''
print(ts.resample('M', kind='period').mean())
'''
2000-01 0.066587
2000-02 -0.240131
2000-03 -0.126769
2000-04 0.387274
Freq: M, dtype: float64
'''
# 降采样是将数据聚合到规整的低频率
rng =pd.date_range('1/1/2000',periods=12,freq='T')
ts=Series(np.arange(12), index=rng)
print(ts)
'''
2000-01-01 00:00:00 0
2000-01-01 00:01:00 1
2000-01-01 00:02:00 2
2000-01-01 00:03:00 3
2000-01-01 00:04:00 4
2000-01-01 00:05:00 5
2000-01-01 00:06:00 6
2000-01-01 00:07:00 7
2000-01-01 00:08:00 8
2000-01-01 00:09:00 9
2000-01-01 00:10:00 10
2000-01-01 00:11:00 11
Freq: T, dtype: int32
'''
# 通过求和的方法将这些数据聚合到“5分钟”块中
print(ts.resample('5min').sum())
'''
2000-01-01 00:00:00 10
2000-01-01 00:05:00 35
2000-01-01 00:10:00 21
Freq: 5T, dtype: int32
'''
# 默认情况下面元的左边界是包含的,因此00:00到00:05区间包含00:05,传入closed='letf'会让区间以左边界闭合
# !!此处跟书中不同,书中是默认包含右边界!!
print(ts.resample('5min', closed='right').sum())
'''
1999-12-31 23:55:00 0
2000-01-01 00:00:00 15
2000-01-01 00:05:00 40
2000-01-01 00:10:00 11
Freq: 5T, dtype: int32
'''
# 时间序列是以各方面左边界的时间戳进行标记的,传入label='right'即可用面元动的右边界对其标记
print(ts.resample('5min',label='right').sum())
'''
2000-01-01 00:05:00 10
2000-01-01 00:10:00 35
2000-01-01 00:15:00 21
Freq: 5T, dtype: int32
'''
# 如果对结果索引做一些位移,如从左边界减去一秒,只需通过loffset设置一个字符串或日期偏移量即可
print(ts.resample('5min',loffset='-1s').sum())
'''
1999-12-31 23:59:59 10
2000-01-01 00:04:59 35
2000-01-01 00:09:59 21
Freq: 5T, dtype: int32
'''
# OHLC重采样,金融领域中有一种无所不在的时间序列聚合方式
# 即计算各面元的四个值,第一个值(开盘)、最后一个值(收盘)、最大值(最高值)、最小值(最低)
# 传入how='ohlc'即可得到一个含有这四种聚合值的DataFrame
print(ts.resample('5min').ohlc())
'''
open high low close
2000-01-01 00:00:00 0 4 0 4
2000-01-01 00:05:00 5 9 5 9
2000-01-01 00:10:00 10 11 10 11
'''
# 通过groupby进行重采样
rng = pd.date_range('1/1/2000', periods=100, freq='D')
ts = Series(np.arange(100), index=rng)
print(ts)
print(ts.groupby(lambda x: x.month).mean())
'''
1 15
2 45
3 75
4 95
dtype: int32
'''
print(ts.groupby(lambda x: x.weekday).mean())
'''
0 47.5
1 48.5
2 49.5
3 50.5
4 51.5
5 49.0
6 50.0
dtype: float64
'''
# 升采样是指将数据从低频率转换到高频率
frame = DataFrame(np.random.randn(2,4),
index=pd.date_range('1/1/2000', periods=2, freq='W-WED'),
columns=['Colorado', 'Texas', 'New York', 'Ohio'])
print(frame[:5])
'''
Colorado Texas New York Ohio
2000-01-05 -0.312261 -1.303667 0.166455 1.113591
2000-01-12 -0.719399 0.860489 0.927483 1.041800
'''
# 将其重采样到日频率,默认会引入缺失值
df_daily = frame.resample('D')
print(df_daily)
'''DatetimeIndexResampler [freq=, axis=0, closed=left, label=left, convention=start, base=0]'''
# 假如想要用前面的周型填充“非星期三”,resample的填充和差值方式跟fillna和reindex的一样
print(frame.resample('D').ffill())
'''
Colorado Texas New York Ohio
2000-01-05 -0.312261 -1.303667 0.166455 1.113591
2000-01-06 -0.312261 -1.303667 0.166455 1.113591
2000-01-07 -0.312261 -1.303667 0.166455 1.113591
2000-01-08 -0.312261 -1.303667 0.166455 1.113591
2000-01-09 -0.312261 -1.303667 0.166455 1.113591
2000-01-10 -0.312261 -1.303667 0.166455 1.113591
2000-01-11 -0.312261 -1.303667 0.166455 1.113591
2000-01-12 -0.719399 0.860489 0.927483 1.041800
'''
# 这里可以只填充指定的时期数(目的是限制前面的观测值持续使用)
print(frame.resample('D').ffill(limit=2))
'''
Colorado Texas New York Ohio
2000-01-05 -0.312261 -1.303667 0.166455 1.113591
2000-01-06 -0.312261 -1.303667 0.166455 1.113591
2000-01-07 -0.312261 -1.303667 0.166455 1.113591
2000-01-08 NaN NaN NaN NaN
2000-01-09 NaN NaN NaN NaN
2000-01-10 NaN NaN NaN NaN
2000-01-11 NaN NaN NaN NaN
2000-01-12 -0.719399 0.860489 0.927483 1.041800
'''
# 新的日期索引完全没有必要跟旧的相交
print(frame.resample('W-THU').ffill())
'''
Colorado Texas New York Ohio
2000-01-06 -0.312261 -1.303667 0.166455 1.113591
2000-01-13 -0.719399 0.860489 0.927483 1.041800
'''
frame = DataFrame(np.random.randn(24,4),
index=pd.period_range('1-2000','12-2001',freq='M'),
columns=['Colorado', 'Texas', 'New York', 'Ohio'])
print(frame[:5])
'''
Colorado Texas New York Ohio
2000-01 0.778957 1.395773 -0.554445 1.233439
2000-02 0.858590 -0.382989 -0.655546 1.364961
2000-03 0.064890 -1.007406 2.427516 -0.147838
2000-04 0.654691 -2.857103 0.011106 -0.549523
2000-05 0.290338 0.226746 1.007994 0.673866
'''
annual_frame = frame.resample('A-DEC').mean()
print(annual_frame)
'''
Colorado Texas New York Ohio
2000 0.406816 -0.262081 -0.186250 0.125175
2001 0.056589 0.340477 0.154083 0.218699
'''
# 升采样稍微麻烦,因为要决定在新的频率中各区间的哪端用于放置原来的值,像asfreq方法
print(annual_frame.resample('Q-DEC').ffill())
'''
Colorado Texas New York Ohio
2000Q1 0.406816 -0.262081 -0.186250 0.125175
2000Q2 0.406816 -0.262081 -0.186250 0.125175
2000Q3 0.406816 -0.262081 -0.186250 0.125175
2000Q4 0.406816 -0.262081 -0.186250 0.125175
2001Q1 0.056589 0.340477 0.154083 0.218699
2001Q2 0.056589 0.340477 0.154083 0.218699
2001Q3 0.056589 0.340477 0.154083 0.218699
2001Q4 0.056589 0.340477 0.154083 0.218699
'''
print(annual_frame.resample('Q-DEC',convention='end').ffill())
'''
Colorado Texas New York Ohio
2000Q4 0.406816 -0.262081 -0.186250 0.125175
2001Q1 0.406816 -0.262081 -0.186250 0.125175
2001Q2 0.406816 -0.262081 -0.186250 0.125175
2001Q3 0.406816 -0.262081 -0.186250 0.125175
2001Q4 0.056589 0.340477 0.154083 0.218699
'''
close_px_all = pd.read_csv('python_data/ch09/stock_px.csv', parse_dates=True,index_col=0)
close_px = close_px_all[['AAPL','MSFT','XOM']]
close_px = close_px.resample('B').ffill()
print(close_px.head())
'''
AAPL MSFT XOM
2003-01-02 7.40 21.11 29.22
2003-01-03 7.45 21.14 29.24
2003-01-06 7.45 21.52 29.96
2003-01-07 7.43 21.93 28.95
2003-01-08 7.28 21.31 28.83
'''
import matplotlib.pyplot as plt
plt.plot(close_px['AAPL'])
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()
# DataFrame调用plot时,时间序列会被绘制在一个subplot上,并有图例说明
close_px.loc['2009'].plot()
# plt.savefig('10-5.png')
plt.show()
# 苹果公司在2011年1月到3月间的每日股价
close_px['AAPL'].loc['01-2011':'03-2011'].plot()
plt.grid(alpha=0.3, linestyle='dashed')
#plt.savefig('10-6.png')
plt.show()
# 季度型频率数据会用季度标记进行格式化
appl_q=close_px['AAPL'].resample('Q-DEC').ffill()
appl_q.loc['2009':].plot()
plt.grid(alpha=0.3, linestyle='dashed')
#plt.savefig('10-7.png')
plt.show()
以下图片对应原书中的图片序号,按代码输出顺序给出(其他段落同样):
图10-4 AAPL每日价格:
# 在移动窗口上计算各种统计函数是一类常见于时间序列的数组变换
# rolling_mean是其中最简单的一个,它接受一个TimeSeries或DataFrame以及一个window(表示期数)
close_px_all = pd.read_csv('python_data/ch09/stock_px.csv', parse_dates=True,index_col=0)
close_px = close_px_all[['AAPL','MSFT','XOM']]
close_px = close_px.resample('B').ffill()
close_px.AAPL.plot()
close_px.AAPL.rolling(250).mean().plot()
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()
appl_std250=close_px.AAPL.rolling(250, min_periods=10).std()
print(appl_std250[5:12])
'''
2003-01-09 NaN
2003-01-10 NaN
2003-01-13 NaN
2003-01-14 NaN
2003-01-15 0.077496
2003-01-16 0.074760
2003-01-17 0.112368
Freq: B, Name: AAPL, dtype: float64
'''
appl_std250.plot()
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()
# 要计算扩展窗口平均,可以将扩展窗口看做一个特殊的窗口,其长度与时间序列一样,但只需一期(或多期)即可计算一个值
# 通过rolling().mean()定义扩展平均
expanding_mean = lambda x: x.rolling(len(x), min_periods=1).mean()
# 对DataFrame调用rolling_mean(以及与之类似的函数)会将转换应用到所有列上
close_px.rolling(60).mean().plot(logy=True)
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()
# 另一种使用固定大小窗口及相等权数观测值得办法是,定义一个衰减因子常量,以便使最近的观测值拥有更大的权数
fig,axes = plt.subplots(2,1, sharex=True, sharey=True, figsize=(12,7))
aapl_px = close_px.AAPL['2005': '2009']
# 对比苹果公司股价的60日移动平均和span=60的指数加权移动平均
ma60 = aapl_px.rolling(60, min_periods=50).mean()
ewma60 = pd.DataFrame.ewm(aapl_px,span=60).mean()
aapl_px.plot(style='k-', ax = axes[0])
ma60.plot(style='k--', ax = axes[0])
aapl_px.plot(style='k-', ax = axes[1])
ewma60.plot(style='k--', ax = axes[1])
axes[0].set_title('Simple MA')
axes[1].set_title('Exponentially-weithed MA')
axes[0].grid(alpha=0.3, linestyle='dashed')
axes[1].grid(alpha=0.3, linestyle='dashed')
plt.show()
# 有些运算需(如相关系数和协方差)需要在两个时间序列上执行
# 通过计算百分数变化并使用rolling_corr的方式得到该结果
spx_px = close_px_all['SPX']
spx_rets = spx_px/spx_px.shift(1)-1
returns = close_px.pct_change()
corr = returns.AAPL.rolling(125, min_periods=100).corr(spx_rets)
corr.plot()
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()
# 计算DataFrame各列与标准普尔500指数的相关系数
corr = returns.rolling(125,min_periods=100).corr(spx_rets)
corr.plot()
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()
图10-12 AAPL6个月的回报与标准普尔500指数的相关系数:
图10-13 3只股票6个月的回报与标准普尔500指数的相关系数:
# rolling apply()函数可以在移动窗口上应用自己设计的数组函数
from scipy.stats import percentileofscore
scroe_at_2percent = lambda x: percentileofscore(x, 0.02)
result = returns.AAPL.rolling(250).apply(scroe_at_2percent)
result.plot()
plt.grid(alpha=0.3, linestyle='dashed')
plt.show()