Python之--时间序列(一)&时间数据分析

1.时间序列

不管在哪个领域中(金融学、经济学、生态学、神经科学、物理学等),时间序列数据都是一种重要的结构化数据形式,在多个时间点观察或者测量到的任何事物都可以形成一段时间序列。很多时间序列是固定频率的,也就是说,数据点是根据某种规律定期出现的(比如每15分钟、每5分钟、每一个月等)。时间序列也可以是不定期的。时间序列数据的意义取决于具体的应用场景,主要有以下几种:

  • 时间戳(timestamp):特定的时刻
  • 固定时期(period):如2019年1月或者2019年全年
  • 时间间隔(interval):由起始和结束时间戳表示。时期也可以看做间隔的特例
  • 实验或过程时间,每个时间点都是相对于特定起始时间的一个度量。例如,从放入烤箱时起,每秒钟饼干的直径。
    下面主要从前3中时间序列来讲解,就目前而言,前三种中时间戳的应用更加常见

2.日期和事件数据类型及工具

Python标准库包含用于时间(time)和日期(date)数据的数据类型,而且还有日历方面的功能,我会主要用到datetime、time以及calendar模块。datetime.datetime(也可以简写为datatime)是用的最多的数据类型:

from datetime import datetime
now = datetime.now()
print(now)
print(now.year)
print(now.month)
print(now.day)

"""
2019-03-25 13:07:37.236060
2019
3
25
"""
  • datetime以毫秒形式存储日期和时间。datetime.timedelta表示两个datetime对象之间的时间差:
delta = datetime(2015,1,7) - datetime(2013,3,4,8,20)
print(delta)
print(delta.days)
print(delta.seconds)


"""
673 days, 15:40:00
673
56400
"""
  • 可以给datetime对象加上或者减去一个或多个timedelta,这样会产生一个新对象:
from datetime import timedelta
start = datetime(2015,1,7)
s = start + timedelta(12)
print(s)

a = start - 2*timedelta(3)
print(a)

"""
2015-01-19 00:00:00
2015-01-01 00:00:00
"""
  • datetime模块中的数据类型参见下表,虽然重点讲解部分是pandas数据类型和高级时间序列处理,但是在实际工作中可定会在Python的其他地方遇到有关datetime的数据类型。
类型说明
date以公历形式存储日历日期(年月日)
time将时间存储为时、分、秒、毫秒
datetime存储日期和时间
timedelta表示两个datetime值之间的差(日、秒、毫秒)

3.字符串和datetime的相互转换

利用str或strftime方法(传入一个格式化字符串),datetime对象和pandas的timestamp对象可以被格式化为字符串:

stamp = datetime(2015,1,7)
print(str(stamp))

"""
2015-01-07 00:00:00
"""
  • 在下表列出了部分的格式化编码。datetime.strptime也可以用这些格式化编码将字符串转化为日期:
代码说明
%Y4位数的年
%y2位数的年
%m2位数的月[01,12]
%d2位数的日[01,31]
%H时(24小时制)[00,23]
%I时(12小时制)[01,12]
%M2位数的分[00,59]
%S秒[00,61](秒60和61用于闰秒)
%w用整数表示的星期几[0(星期天),6]
value = '2001-01-07'
v = datetime.strptime(value,'%Y-%m-%d')
print(v)

values = ['7/1/2011','9/3/2019']
vs = [datetime.strptime(value,'%m/%d/%Y') for value in values]
print(vs)

"""
2001-01-07 00:00:00
[datetime.datetime(2011, 7, 1, 0, 0), datetime.datetime(2019, 9, 3, 0, 0)]
"""
  • datetime.strptime是通过已知格式进行日期解析的最佳方式。但是每次都要编写格式定义是很麻烦的事情,尤其是对于一些常见的日期格式。这种情况下,你可以用dateutil这个第三方库中的parser.parse方法:
from dateutil.parser import parse
print(parse('2017/01/07'))

"""
2017-01-07 00:00:00
"""
  • dateutil可以解析几乎所有人类能够理解的日期表示形式(但是中文不行):
    注意:dateutil.parser是一个使用但是不完美的工具。比如说,他会把一些原本不是日期的字符串认作是日期(比如“12”会被解析为今年今月的12日)
print(parse('Jan 31 1999 10:20 PM'))
print(parse('Jan 31 1999 10:20 PM'))
print(parse('12'))

"""
1999-01-31 22:20:00
2019-03-12 00:00:00
"""
  • 在国际通用的格式中,日常出现在月的前面,传入dayfirst =Ture即可解决这个问题(在后续的版本中不加也是可以的):
print(parse('7/1/2015',dayfirst = True))

"""
2015-01-07 00:00:00
"""
  • pandas通常是用于处理成组日期的,不管这些日期是DataFrame的轴索引还是列。to_datetime方法可以解析多种不同的日期表示形式。对标准日期格式(如ISO8601)的解析非常快捷。
print(values)
print(pd.to_datetime(values))

"""
['7/1/2011', '9/3/2019']
DatetimeIndex(['2011-07-01', '2019-09-03'], dtype='datetime64[ns]', freq=None)
"""
  • 它可以处理缺失值(None,空字符串等):
idx = pd.to_datetime(values + [None])
print(idx)
print(pd.isnull(idx))

"""
DatetimeIndex(['2011-07-01', '2019-09-03', 'NaT'], dtype='datetime64[ns]', freq=None)
[False False  True]
"""
  • NaT(Not a Time)是pandas中时间戳数据的NA值

4.时间序列基础

pandas最基本的时间序列类型就是以时间戳(通常以Python字符串或者datatime对象表示)为索引的Series:

from datetime import datetime
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 = pd.Series(np.random.randn(6),index=dates)
print(ts)

"""
2011-01-02    1.598238
2011-01-05    0.333948
2011-01-07   -0.347980
2011-01-08   -1.603143
2011-01-10   -1.080838
2011-01-12   -0.655313
dtype: float64
"""
  • 这些datetime对象实际上是被放在一个DatetimeIndex中的。现在,变量ts就成为一个TimeSeries了:
print(type(ts))
print(ts.index)

"""
<class 'pandas.core.series.Series'>
DatetimeIndex(['2011-01-02', '2011-01-05', '2011-01-07', '2011-01-08',
               '2011-01-10', '2011-01-12'],
              dtype='datetime64[ns]', freq=None)
"""

注意:没必要显示使用TimeSeries的构造函数。当创建一个带有DatetimeIndex的Series时,pandas就会知道该对象是一个时间序列。

  • 跟其他Series一样,不同的索引的时间序列之间的算术运算会自动按日期对齐:
print(ts + ts[::2])

"""
2011-01-02    3.196476
2011-01-05         NaN
2011-01-07   -0.695960
2011-01-08         NaN
2011-01-10   -2.161677
2011-01-12         NaN
dtype: float64
"""
  • pandas用Numpy的datetime64数据类型以纳秒形式存储时间戳:
print(ts.index.dtype)

"""
datetime64[ns]
"""
  • DateTimeIndex中的各个标量值是pandas的Timestamp对象:
stamp = ts.index[0]
print(stamp)
"""
2011-01-02 00:00:00
"""
  • 只要有需要,TimeStamp可以随时自动转换为datetime对象,此外,它还可以存储频率信息,且知道如何执行时区转换以及其他操作。

5.索引、选取、子集构造

由于TimeSeries是Series的一个子类,所以在索引以及数据选取方面他们的行为是一样的:

stamp = ts.index[2]
print(ts[stamp])

"""
-0.34797996475619797
"""
  • 还有一种更为方便的用法,传入一个可以被解释为日期的字符串。
# 注意格式,月/日/年
ts['1/07/2011']

"""
-0.34797996475619797
"""
  • 对于较长的时间序列,只需要传入“年”或者“年月”即可轻松选取数据的切片:
longer_ts = pd.Series(np.random.randn(1000),
                      index=pd.date_range('1/1/2000',periods=1000))
print(longer_ts['2001'])

"""
2001-01-01   -0.354147
2001-01-02   -0.411002
                ...   
2001-12-30   -0.777096
2001-12-31   -1.450070
Freq: D, Length: 365, dtype: float64                
"""

print(longer_ts['2001-5'])

"""
2001-05-01    0.624091
2001-05-02   -0.702856
               ...  
2001-05-30   -0.496207
2001-05-31    1.000048
Freq: D, Length: 31, dtype: float64               
"""
  • 通过日期进行切片的方式支队规则Series有效:
print(ts[datetime(2011,1,7):])

"""
2011-01-07   -0.347980
2011-01-08   -1.603143
2011-01-10   -1.080838
2011-01-12   -0.655313
dtype: float64
"""
  • 由于大部分时间序列数据都是按照时间先后排序的,因此你也可以用不存在与该时间序列中的时间戳对其进行切片(范围查询):
print(ts)

"""
2011-01-02    1.598238
2011-01-05    0.333948
2011-01-07   -0.347980
2011-01-08   -1.603143
2011-01-10   -1.080838
2011-01-12   -0.655313
dtype: float64
"""
print(ts['1/7/2011':'1/10/2011'])

"""
2011-01-07   -0.347980
2011-01-08   -1.603143
2011-01-10   -1.080838
dtype: float64
"""
  • 跟之前一样,这里可以传入字符串日期、datetime或者Timestamp。注意,这样切片所产生的是源时间序列的视图,跟numpy数组的切片运算是一样的,此外,还有一个等价的实例方法也可以截取两个日期之间TimeSeries:
print(ts.truncate(after = '1/7/2011'))
"""
2011-01-02    1.598238
2011-01-05    0.333948
2011-01-07   -0.347980
dtype: float64
"""
  • 上面这些操作对DataFrame也有效,例如,对DataFrame的行进行索引:
dates = pd.date_range('1/1/2001',periods=100,freq='W-WED')
long_df = pd.DataFrame(np.random.randn(100,4),
                      index=dates,
                      columns=['A','B','C','D'])
print(long_df.ix['4-2002'])

"""
                   A         B         C         D
2002-04-03  0.382645  0.297246  0.259205 -0.355514
2002-04-10  2.171299 -0.234009 -2.047130 -0.568277
2002-04-17  0.436831  0.856942  0.485755  0.490481
2002-04-24  0.220224 -0.485198 -0.263579 -0.562568
"""

6.带有重复索引的时间序列

在某些应用场景中,可能会存在多个观测数据罗在同一个时间点上的情况,下面就一个例子:

dates = pd.DatetimeIndex(['1/1/2001','1/2/2001','1/2/2001','1/2/2001',
                     '1/2/2001','1/3/2001'])
dup_ts = pd.Series(np.arange(6),index=dates)
print(dup_ts)

"""
2001-01-01    0
2001-01-02    1
2001-01-02    2
2001-01-02    3
2001-01-02    4
2001-01-03    5
dtype: int64
"""
"""
  • 通过检查索引的is_unique属性,就可以知道它是不是唯一的:
dup_ts.index.is_unique

"""
False
"""
  • 对这个时间序列进行索引,要么产生标量值,要么产生切片,具体要看所选的时间点是否重复:
dup_ts['1/3/2001']# 不重复

"""
5
"""

dup_ts['1/2/2001']# 重复

"""
2001-01-02    1
2001-01-02    2
2001-01-02    3
2001-01-02    4
dtype: int64
"""
  • 假设你想要对具有非唯一时间戳的数据进行聚合。一个办法是使用GroupBy,并传入level = 0(索引的唯一一层):
grouped = dup_ts.groupby(level= 0)
print(grouped.mean())

"""
2001-01-01    0.0
2001-01-02    2.5
2001-01-03    5.0
dtype: float64
"""
print(grouped.count())

"""
2001-01-01    1
2001-01-02    4
2001-01-03    1
dtype: int64
"""
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值