Prophet Docs

走马观花过下,方便后面查阅

安装:https://facebook.github.io/prophet/docs/installation.html

# 一些warning不要了
import warnings
warnings.filterwarnings('ignore')

1. 先跑个demo

用法:先创建Prophet对象,再调用fitpredict

输入数据:含有dsy列的DataFrame

y: 必须是数值,要预测的值
ds: datestamp, 日期(YYYY-MM-DD)或者时间戳(YYYY-MM-DD HH:MM:SS)

下面使用数据集peyton_manning试一下。

1.1 下载数据:

!curl -o example_wp_log_peyton_manning.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_wp_log_peyton_manning.csv
% Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100 86845  100 86845    0     0  14309      0  0:00:06  0:00:06 --:--:--  9945

1.2 导入数据

import pandas as pd
from fbprophet import Prophet
df = pd.read_csv('example_wp_log_peyton_manning.csv')
df.head()
ds y
0 2007-12-10 9.590761
1 2007-12-11 8.519590
2 2007-12-12 8.183677
3 2007-12-13 8.072467
4 2007-12-14 7.893572
df.ds
0       2007-12-10
1       2007-12-11
2       2007-12-12
3       2007-12-13
4       2007-12-14
           ...    
2900    2016-01-16
2901    2016-01-17
2902    2016-01-18
2903    2016-01-19
2904    2016-01-20
Name: ds, Length: 2905, dtype: object

可以看到ds从2007-12-10到2016-01-20

1.3 创建一个Prophet实例进行fit

m = Prophet()
m.fit(df)
INFO:numexpr.utils:NumExpr defaulting to 8 threads.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.





<fbprophet.forecaster.Prophet at 0x10546f9d0>

1.4 扩展ds列

使用Prophet.make_future_dataframe(看看R API的文档)扩展ds列,扩展后包含有历史的日期和扩展日期
也可以扩展有时分秒的时间

future = m.make_future_dataframe(periods=365)
future.ds
0      2007-12-10
1      2007-12-11
2      2007-12-12
3      2007-12-13
4      2007-12-14
          ...    
3265   2017-01-15
3266   2017-01-16
3267   2017-01-17
3268   2017-01-18
3269   2017-01-19
Name: ds, Length: 3270, dtype: datetime64[ns]

可以看到future.ds从2007-12-10到2017-01-19,包含了历史的日期(2007-12-10到2016-01-20)和新扩展的日期(2016-01-21到2017-01-19)

1.5 预测

预测结果是一个新的DataFrame对象,并且包含了预测值yhat,各种成分及预测区间[xxx_lower, xxx_upper]

forecast = m.predict(future)
forecast.tail()
image.png

画图看看预测效果

# 这里不用变量接收会画两次
fig1 = m.plot(forecast)
image.png

看看预测的各种成分,趋势、年度季节性、星期季节性、以及节假日等

fig2 = m.plot_components(forecast)
image.png

可以使用plotly画出交互图

from fbprophet.plot import plot_plotly
import plotly.offline as py
py.init_notebook_mode()

fig = plot_plotly(m, forecast)  # This returns a plotly Figure
py.iplot(fig)

2. 增长/下降预测

2.1 预测增长

默认情况下,Prophet使用线性模型进行预测
预测增长时,通常会有一个最大值(承载容量)(默认最小值是0)
Prophet可以指定一个承载容量,使用逻辑斯蒂增长趋势模型(Logistic growth trend model)进行预测。
下面使用维基百科页面R (programming language)pvlog后举例:

2.1.1 下载数据

!curl -o example_wp_log_R.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_wp_log_R.csv
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100 85609  100 85609    0     0  10412      0  0:00:08  0:00:08 --:--:-- 14715 0:00:12  5486

2.1.2 读取数据并指定最大容量(cap列)

df = pd.read_csv('example_wp_log_R.csv')
df['cap'] = 8.5

注:每一行都需要指定capcap不一定是个固定值,cap也可以是一个增长序列(随着y的增长而增长)

2.1.3 使用参数growth=logistic指定logistic增长,再fit

m = Prophet(growth='logistic')
m.fit(df)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.





<fbprophet.forecaster.Prophet at 0x1a1dfd6b50>

2.1.4 扩展ds列

和之前不同的是,这次需要指定最大增长量
这里指定为一个定值,并预测未来三年的log(pv)

future = m.make_future_dataframe(periods=3*365)
future['cap'] = 8.5
fcst = m.predict(future)
fig = m.plot(fcst)
image.png

2.2 下降预测

默认最小值为0,也可以通过floor列单独指定
使用逻辑斯蒂增长趋势模型预测减少的时候,必须指定最大值

df['y'] = 10 - df['y']
df['cap'] = 6
df['floor'] = 1.5
future['cap'] = 6
future['floor'] = 1.5
m = Prophet(growth='logistic')
m.fit(df)
fcst = m.predict(future)
fig = m.plot(fcst)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

这里不指定最小值再试下

# 上面已经修改过df了,重新加载数据
df = pd.read_csv('example_wp_log_R.csv')
# 最大为8.5
df['y'] = 8.5 - df['y']
df['cap'] = 4
#df['floor'] = 1.5
future['cap'] = 4
#future['floor'] = 1.5
m = Prophet(growth='logistic')
m.fit(df)
fcst = m.predict(future)
fig = m.plot(fcst)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.

image.png

可以看到默认最小值为0

3. 趋势中的变化点

前面的几个例子中,可以看到真实的趋势中会存在一些突变的点
Prophet默认会自动检测到这些点并调整趋势,但是,自动调整出现问题时(比如过拟合),也可以手动调整

3.1 自动检测变化点

  1. 通过大量的速率改变的点检测变化点
  2. 对这些点做稀疏先验(sparse prior)

实际上有很多可能的变化点,但会尽可能少用

在数据集example_wp_log_peyton_manning.csv上,默认情况下会检测出25个变化点,这些点均匀的分布在前80%的时间序列中
就是下面这些竖线:

df = pd.read_csv('example_wp_log_peyton_manning.csv')
m = Prophet()
m.fit(df)
future = m.make_future_dataframe(periods=365)
forecast = m.predict(future)
fig = m.plot(forecast)
for cp in m.changepoints:
    plt.axvline(cp, c='gray', ls='--', lw=2)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.

image.png

因为稀疏先验,大部分的变化点并没有用到
看看每个变化点的速率变化图:

deltas = m.params['delta'].mean(0)
fig = plt.figure(facecolor='w', figsize=(10, 6))
ax = fig.add_subplot(111)
ax.bar(range(len(deltas)), deltas, facecolor='#0072B2', edgecolor='#0072B2')
ax.grid(True, which='major', c='gray', ls='-', lw=1, alpha=0.2)
ax.set_ylabel('Rate change')
ax.set_xlabel('Potential changepoint')
fig.tight_layout()

image.png

变化点的数量可以通过参数n_changepoints指定,但最好还是通过调整正则化来修改
下面看看比较明显的变化点:

from fbprophet.plot import add_changepoints_to_plot
fig = m.plot(forecast)
a = add_changepoints_to_plot(fig.gca(), m, forecast)

image.png

默认情况下,为了有足够的长度预测未来趋势和防止过拟合,会在时间序列的前80%推断变化点,但是也可以通过参数changepoint_range来修改变化点所在范围,比如m = Prophet(changepoint_range=0.9)改为前90%

3.2 调整趋势灵活性

当趋势出现过拟合或者欠拟合的情况下,可以通过参数changepoint_prior_scale调整稀疏先验的程度,默认为0.05
该参数值越大,则趋势越灵活

增大灵活性

m = Prophet(changepoint_prior_scale=0.9)
forecast = m.fit(df).predict(future)
fig = m.plot(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

减少灵活性

m = Prophet(changepoint_prior_scale=0.001)
forecast = m.fit(df).predict(future)
fig = m.plot(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

3.3 指定变化点的位置

也可以通过参数changepoints手动指定位置,只有指定的这些点可以有速率变化

m = Prophet(changepoints=['2014-01-01'])
forecast = m.fit(df).predict(future)
fig = m.plot(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

4. 季节性、假期效应和回归因子

4.1 假期和特殊事件建模

要给假期或者其他重复事件建模,就需要创建一个包含holidayds列的DataFrame
需要包含过去和将来所有的特殊日子,如果这些特殊日子没有出现在将来(要预测的日期),那预测就不会用到

通过lower_window列和upper_window列来扩展假期,比如双十一扩展为11.1~11.20
通过prior_scale列来指定每个假期的prior scale

下面创建一个包含了Peyton Manning’s所有决赛日期的DataFrame

playoffs = pd.DataFrame({
  'holiday': 'playoff',
  'ds': pd.to_datetime(['2008-01-13', '2009-01-03', '2010-01-16',
                        '2010-01-24', '2010-02-07', '2011-01-08',
                        '2013-01-12', '2014-01-12', '2014-01-19',
                        '2014-02-02', '2015-01-11', '2016-01-17',
                        '2016-01-24', '2016-02-07']),
  'lower_window': 0,
  'upper_window': 1,
})
superbowls = pd.DataFrame({
  'holiday': 'superbowl',
  'ds': pd.to_datetime(['2010-02-07', '2014-02-02', '2016-02-07']),
  'lower_window': 0,
  'upper_window': 1,
})
holidays = pd.concat((playoffs, superbowls))

上面superbowl的日期也包含在playoff的日期中,也就是superbowl日期的影响会有个叠加效应

使用下创建好的holidays:

m = Prophet(holidays=holidays)
forecast = m.fit(df).predict(future)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.

可以通过forecast看看假期效应

forecast[(forecast['playoff'] + forecast['superbowl']).abs() > 0][
        ['ds', 'playoff', 'superbowl']][-10:]

image.png

看看假期效应在图上的显示,playoff日期有高峰,superbowl日期有更明显的高峰:

fig = m.plot_components(forecast)
image.png

可以使用

from fbprophet.plot import plot_forecast_component
plot_forecast_component(m, forecast, 'superbowl')

对假期单独画图

4.2 内置假期

可以通过add_country_holidays使用内置假期
通过模型的train_holiday_names方法查看哪些假期

m = Prophet(holidays=holidays)
m.add_country_holidays(country_name='CN')
m.fit(df)

m.train_holiday_names
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.





0                 playoff
1               superbowl
2          New Year's Day
3        Chinese New Year
4       Tomb-Sweeping Day
5               Labor Day
6    Dragon Boat Festival
7     Mid-Autumn Festival
8            National Day
dtype: object

所有的假期都在holidays包中提供
假期的日期范围可以这个脚本替换: generate_holidays_file.py

再画个图:

m = Prophet(holidays=holidays)
m.add_country_holidays(country_name='US')
m.fit(df)

forecast = m.predict(future)
fig = m.plot_components(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

4.3 季节性的傅里叶级数(Fourier Order for Seasonalities)

季节性是用傅里叶和(Fourier sum)估算的

这段直接贴原文吧:

Seasonalities are estimated using a partial Fourier sum. See the paper for complete details, and this figure on Wikipedia for an illustration of how a partial Fourier sum can approximate an aribtrary periodic signal. The number of terms in the partial sum (the order) is a parameter that determines how quickly the seasonality can change.

这里仍然以Peyton Manning的数据为例,年度季节性的傅里叶级数默认是10,画下图:

df = pd.read_csv('example_wp_log_peyton_manning.csv')
from fbprophet.plot import plot_yearly
m = Prophet().fit(df)
a = plot_yearly(m)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

这个默认值大多数是没问题的,但是季节性可能有更高频率的变化,并且通常没有这么平滑,这时可以增加这个值
增加这个值也可能导致过拟合, N Fourier terms corresponds to 2N variables used for modeling the cycle
这里增加到20,画图看看

from fbprophet.plot import plot_yearly
m = Prophet(yearly_seasonality=20).fit(df)
a = plot_yearly(m)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

4.4 自定义季节性

  1. 时间序列超过两个周期时,Prophet默认训练星期和年的季节性
  2. sub-daily的时间序列时,会训练每天的季节性
  3. 可以使用函数add_seasonality添加小时/月/季度等其他季节性

函数add_seasonality的参数:

  1. name 哪种周期
  2. period 季节性的周期
  3. fourier_order 季节性的傅里叶级数
  4. prior_scale 可选参数,下面会说

默认情况下,周的季节性傅里叶级数为3,年的季节性傅里叶级数为10

仍然使用Peyton Manning数据集,将每周的季节性替换为每月的季节性(period=30.5),画个图:

m = Prophet(weekly_seasonality=False)
m.add_seasonality(name='monthly', period=30.5, fourier_order=5)
forecast = m.fit(df).predict(future)
fig = m.plot_components(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

4.5 依赖于其他因素的季节性

有时候季节性依赖于其他一些因素,比如每周的季节性在夏天和其他季节表现不一致,每天的季节性在周末和周内表现不一致
这种季节性可以使用带条件的季节性训练模型(These types of seasonalities can be modeled using conditional seasonalities.)

在前面的Peyton Manning例子中,在一年中默认每周的季节性表现是一致的,但是可能希望每周的季节性在淡季和旺季(每周末有比赛)表现不一致
我们可以使用带条件的季节性为淡季和旺季单独构建每周的季节性

先增加一列布尔类型的数据,来表示日期在淡季还是旺季:

def is_nfl_season(ds):
    date = pd.to_datetime(ds)
    return (date.month > 8 or date.month < 2)

df['on_season'] = df['ds'].apply(is_nfl_season)
df['off_season'] = ~df['ds'].apply(is_nfl_season)

接着禁用内置的每周季节性,使用淡季的周季节性和旺季的周季节性代替
因此,只有condition_name列为True的时候季节性才有日期
在预测的DataFrame上,也要做同样的操作

m = Prophet(weekly_seasonality=False)
m.add_seasonality(name='weekly_on_season', period=7, fourier_order=3, condition_name='on_season')
m.add_seasonality(name='weekly_off_season', period=7, fourier_order=3, condition_name='off_season')

future['on_season'] = future['ds'].apply(is_nfl_season)
future['off_season'] = ~future['ds'].apply(is_nfl_season)
forecast = m.fit(df).predict(future)
fig = m.plot_components(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

从图中可以看到,在旺季的时候每周末都会打球,周日和周一都有大幅度增长,但在淡季则完全没有。

4.6 假期和季节性的prior scale

如果发现假期过拟合,可以设置参数holidays_prior_scale调整假期的prior scale使之平滑
这个参数默认是10,减少可以限制假期效果

m = Prophet(holidays=holidays, holidays_prior_scale=0.05).fit(df)
forecast = m.predict(future)
forecast[(forecast['playoff'] + forecast['superbowl']).abs() > 0][
    ['ds', 'playoff', 'superbowl']][-10:]
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

可以看到,比起之前假期效应被减弱了,特别是在观看最少的superbowls

可以用下面的方式设置每周季节性的prior_scale

m = Prophet()
m.add_seasonality(
    name='weekly', period=7, fourier_order=3, prior_scale=0.1)
<fbprophet.forecaster.Prophet at 0x1a21d06b90>

4.7 额外的回归特征

可以使用函数add_regressor将其他回归特征添加到模型的线性部分
训练和预测的数据集上都需要包含这些回归特征的值

下面,为NFL赛季的每周日添加这样一个回归特征,再画图看看这个特征的效果

def nfl_sunday(ds):
    date = pd.to_datetime(ds)
    if date.weekday() == 6 and (date.month > 8 or date.month < 2):
        return 1
    else:
        return 0
df['nfl_sunday'] = df['ds'].apply(nfl_sunday)

m = Prophet()
m.add_regressor('nfl_sunday')
m.fit(df)

future['nfl_sunday'] = future['ds'].apply(nfl_sunday)

forecast = m.predict(future)
fig = m.plot_components(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

也可以使用前面说过的holidays的接口,通过创建一个过去和未来的这些周日的list来处理NFL赛季周日的这种情况
函数add_regressor为定义额外的线性回归提供了一个更加通用的接口

接下来的给出原文吧:

NFL Sundays could also have been handled using the “holidays” interface described above, by creating a list of past and future NFL Sundays. The add_regressor function provides a more general interface for defining extra linear regressors, and in particular does not require that the regressor be a binary indicator. Another time series could be used as a regressor, although its future values would have to be known.

This notebook shows an example of using weather factors as extra regressors in a forecast of bicycle usage, and provides an excellent illustration of how other time series can be included as extra regressors.

The add_regressor function has optional arguments for specifying the prior scale (holiday prior scale is used by default) and whether or not the regressor is standardized - see the docstring with help(Prophet.add_regressor) in Python and ?add_regressor in R. Note that regressors must be added prior to model fitting.

The extra regressor must be known for both the history and for future dates. It thus must either be something that has known future values (such as nfl_sunday), or something that has separately been forecasted elsewhere. Prophet will also raise an error if the regressor is constant throughout the history, since there is nothing to fit from it.

Extra regressors are put in the linear component of the model, so the underlying model is that the time series depends on the extra regressor as either an additive or multiplicative factor (see the next section for multiplicativity).

5. 乘法季节性

默认情况下Prophet训练的是加法季节性,这种方式预测值是季节性影响加上趋势得到的

对于航空乘客数的变化,加法季节性就会有问题:

先下载数据集

!curl -o example_air_passengers.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_air_passengers.csv
    
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100  2165  100  2165    0     0   1980      0  0:00:01  0:00:01 --:--:--  1982
df = pd.read_csv('example_air_passengers.csv')
m = Prophet()
m.fit(df)
future = m.make_future_dataframe(50, freq='MS')
forecast = m.predict(future)
fig = m.plot(forecast)
INFO:fbprophet:Disabling weekly seasonality. Run prophet with weekly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

这个时间序列中在每一年中明显存在周期,但是预测的季节性在开始时间太大,在结束时间太小
这里的季节性不是一个恒定的加数,而是随着趋势在增长

可以通过设置参数seasonality_mode='multiplicative'来为乘法季节性进行建模:

m = Prophet(seasonality_mode='multiplicative')
m.fit(df)
forecast = m.predict(future)
fig = m.plot(forecast)

fig2 = m.plot_components(forecast)
INFO:fbprophet:Disabling weekly seasonality. Run prophet with weekly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

image.png

因为设置了参数seasonality_mode='multiplicative', 所以假期效果也会使用乘法建模
可以通过参数seasonality_mode给模型设置季节性模式,如果要修改单个季节性的模式,可以通过设置mode='additive'或者mode='multiplicative'来覆盖原有设置

看这里,模型设置的季节性为乘法模式,季度的季节性和另一个回归因子使用加法模式覆盖了原来的乘法模式:

m = Prophet(seasonality_mode='multiplicative')
m.add_seasonality('quarterly', period=91.25, fourier_order=8, mode='additive')
m.add_regressor('regressor', mode='additive')
<fbprophet.forecaster.Prophet at 0x1a222e2150>

6. 预测区间

默认情况下Prophet会返回一个预测区间
三部分不确定性:趋势和季节性的不确定性、观测结果的噪声影响

6.1 趋势的不确定

对于预测结果的不确定性,影响最大的就是未来趋势的变化
未来趋势的变化时没有办法准确预知的,只能尽可能合理的预测,这里需要假定未来和历史有相似的趋势
特别是,假设未来和过去的趋势变化频率和大小是一致的,从而预测未来的趋势变化,进而计算出预测范围。这个假设虽然合理却不一定正确,所以预测的结果范围可能也不会完全准确

使用这种计算不确定性的方法时,变化速率越灵活,预测的结果范围也越大,变化范围的灵活性可以通过参数changepoint_prior_scale设置(越大越灵活),这个预测范围也可以用来判定是否过拟合

可以使用interval_width来设置预测区间的宽度(默认80%):

forecast = Prophet(interval_width=0.95).fit(df).predict(future)
INFO:fbprophet:Disabling weekly seasonality. Run prophet with weekly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.

6.2 季节性的不确定

默认Prophet只会返回趋势的不确定和观测结果的噪声影响
要想得到季节性的不确定性,必须通过全贝叶斯采样(full Bayesian sampling),这里需要设置参数mcmc.samples(默认0)来实现

Peyton Manning数据集的前六个月数据为例,这里把最大后验估计(Maximum-a-Posteriori (MAP) Estimation)替换成了MCMC采样,并且将观测结果的时间从几秒延长到了几分钟
如果做了全采样,就能画图看到季节性的不确定性了

原文:

This replaces the typical MAP estimation with MCMC sampling, and can take much longer depending on how many observations there are - expect several minutes instead of several seconds. If you do full sampling, then you will see the uncertainty in seasonal components when you plot them:

df = pd.read_csv('example_wp_log_peyton_manning.csv')
df = df.loc[:180,]  # Limit to first six months
m = Prophet()
m.fit(df)
future = m.make_future_dataframe(periods=60)
m = Prophet(mcmc_samples=300)
forecast = m.fit(df).predict(future)

fig = m.plot_components(forecast)
INFO:fbprophet:Disabling yearly seasonality. Run prophet with yearly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
INFO:fbprophet:Disabling yearly seasonality. Run prophet with yearly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

可以使用函数m.predictive_samples(future)来访问原始的后验预测样本(posterior predictive samples)

在Windows下使用PyStan做MCMC采样的时候巨慢,做MCMC采样的时候,如果使用Windows,最好使用R,如果用Linux,最好使用Python

7. 异常值(Outliers)

异常值主要有两种方式影响预测结果

下面用一个有错误数据的维基百科某词条的PV数据来试试

先下载数据集

!curl -o example_wp_log_R_outliers1.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_wp_log_R_outliers1.csv
    
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100 80366  100 80366    0     0  15180      0  0:00:05  0:00:05 --:--:-- 19630
df = pd.read_csv('example_wp_log_R_outliers1.csv')
m = Prophet()
m.fit(df)
future = m.make_future_dataframe(periods=1096)
forecast = m.predict(future)
fig = m.plot(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

这个预测的趋势看起来没啥问题,但这个预测的区间太特么大了
Prophet可以处理历史数据的异常值,但只是把这些异常值和趋势变化拟合在一起,模型会认为未来的趋势变化也有相同的程度

最好的处理异常值的方式就是干掉它们(Prophet可以处理缺失数据),如果将一些日期的历史数据设置为NA,但未来的日期中存在,Prophet照样能给出这些日期的预测值

df.loc[(df['ds'] > '2010-01-01') & (df['ds'] < '2011-01-01'), 'y'] = None
model = Prophet().fit(df)
fig = model.plot(model.predict(future))
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

上面的例子中异常值扰乱了不确定性估计,却并没有影响到预测结果

有时候又不是酱紫,下面添加了一些新的异常值:

!curl -o example_wp_log_R_outliers2.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_wp_log_R_outliers2.csv
    
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100 85194  100 85194    0     0   8389      0  0:00:10  0:00:10 --:--:-- 128326
df = pd.read_csv('example_wp_log_R_outliers2.csv')
m = Prophet()
m.fit(df)
future = m.make_future_dataframe(periods=1096)
forecast = m.predict(future)
fig = m.plot(forecast)
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

2015年6月这里有一堆极端异常值扰乱了季节性估计,这种影响会在未来的预测中会一直存在,要消除这种影响,还是得干掉这些数据:

df.loc[(df['ds'] > '2015-06-01') & (df['ds'] < '2015-06-30'), 'y'] = None
m = Prophet().fit(df)
fig = m.plot(m.predict(future))
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

8. Non-Daily日期(Non-Daily Data)

8.1 Sub-daily日期

Prophet可以为sub-daily的时间序列预测结果,时间戳格式要求为YYYY-MM-DD HH:MM:SS,这种情况下,每日季节性就会自动拟合

下面用一个5分钟间隔的温度数据试一下
下载数据集

!curl -o example_yosemite_temps.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_yosemite_temps.csv
    
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100  451k  100  451k    0     0   6086      0  0:01:15  0:01:15 --:--:--  6972 8312 0:01:23  0:00:18  0:01:05  605424  0:00:25  0:00:59  480915  0:00:38  0:00:37  4779 0:01:17  0:00:50  0:00:27  4239:01:14  0:00:02  7468
df = pd.read_csv('example_yosemite_temps.csv')
m = Prophet(changepoint_prior_scale=0.01).fit(df)
future = m.make_future_dataframe(periods=300, freq='H')
fcst = m.predict(future)
fig = m.plot(fcst)
INFO:fbprophet:Disabling yearly seasonality. Run prophet with yearly_seasonality=True to override this.
image.png

看下每日季节性吧:

fig = m.plot_components(fcst)
image.png

8.2 有规律间隔的数据

假设上面的数据集只有每天凌晨到早上六点的数据

df2 = df.copy()
df2['ds'] = pd.to_datetime(df2['ds'])
df2 = df2[df2['ds'].dt.hour < 6]
m = Prophet().fit(df2)
future = m.make_future_dataframe(periods=300, freq='H')
fcst = m.predict(future)
fig = m.plot(fcst)
INFO:fbprophet:Disabling yearly seasonality. Run prophet with yearly_seasonality=True to override this.
image.png

这次预测的很差,未来波动实在太大了。因为训练数据只有零点到六点的数据,所以在每天的其他时间,每日季节性就不受约束了
要解决这个问题,还是要把预测的时间控制在零点到六点,其他时间干掉

future2 = future.copy()
future2 = future2[future2['ds'].dt.hour < 6]
fcst = m.predict(future2)
fig = m.plot(fcst)
image.png

对于其他类似的,也需要这样处理,比如,如果历史记录只包含周内,那么只能对周内进行预测,要是预测周末就会垮了

8.3 月数据

Prophet可以拟合月数据,但是,基础模型是时间连续的,如果使用月数据训练模型,但是进行每日预测,结果就会很诡异。

这里,预测下美国未来10年的零售额:

下载数据集

!curl -o example_retail_sales.csv https://raw.githubusercontent.com/facebook/prophet/master/examples/example_retail_sales.csv
    
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100  5279  100  5279    0     0   3207      0  0:00:01  0:00:01 --:--:--  3207
df = pd.read_csv('example_retail_sales.csv')
m = Prophet(seasonality_mode='multiplicative').fit(df)
future = m.make_future_dataframe(periods=3652)
fcst = m.predict(future)
fig = m.plot(fcst)
INFO:fbprophet:Disabling weekly seasonality. Run prophet with weekly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

跟上面的问题类似,拟合的是年季节性,训练数据只有每月第一天的数据,因此其他天的季节性是不可识别和过拟合的
通过MCMC看看这种不确定的季节性:

m = Prophet(seasonality_mode='multiplicative', mcmc_samples=300).fit(df)
fcst = m.predict(future)
fig = m.plot_components(fcst)
INFO:fbprophet:Disabling weekly seasonality. Run prophet with weekly_seasonality=True to override this.
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
image.png

可以发现,每月初的季节不确定性很低
可以通过函数make_future_dataframe传入freq参数来实现只对月数据的预测

future = m.make_future_dataframe(periods=120, freq='M')
fcst = m.predict(future)
fig = m.plot(fcst)
image.png

9. 诊断

Prophet有交叉验证功能
具体做法是通过在历史数据中选择一些截断点,对于这些截断点,只使用这些点之前的数据来拟合模型,然后比较真实值和预测值

下面这个图显示了Peyton-Manning数据集上模拟的历史预测,模型使用前五年的数据训练,预测后一年的数据

from fbprophet import Prophet
import pandas as pd
from matplotlib import pyplot as plt
df = pd.read_csv('example_wp_log_peyton_manning.csv')
m = Prophet()
m.fit(df)
future = m.make_future_dataframe(periods=366)

from fbprophet.diagnostics import cross_validation
df_cv = cross_validation(
    m, '365 days', initial='1825 days', period='365 days')
cutoff = df_cv['cutoff'].unique()[0]
df_cv = df_cv[df_cv['cutoff'].values == cutoff]

fig = plt.figure(facecolor='w', figsize=(10, 6))
ax = fig.add_subplot(111)
ax.plot(m.history['ds'].values, m.history['y'], 'k.')
ax.plot(df_cv['ds'].values, df_cv['yhat'], ls='-', c='#0072B2')
ax.fill_between(df_cv['ds'].values, df_cv['yhat_lower'],
                df_cv['yhat_upper'], color='#0072B2',
                alpha=0.2)
ax.axvline(x=pd.to_datetime(cutoff), c='gray', lw=4, alpha=0.5)
ax.set_ylabel('y')
ax.set_xlabel('ds')
ax.text(x=pd.to_datetime('2010-01-01'),y=12, s='Initial', color='black',
       fontsize=16, fontweight='bold', alpha=0.8)
ax.text(x=pd.to_datetime('2012-08-01'),y=12, s='Cutoff', color='black',
       fontsize=16, fontweight='bold', alpha=0.8)
ax.axvline(x=pd.to_datetime(cutoff) + pd.Timedelta('365 days'), c='gray', lw=4,
           alpha=0.5, ls='--')
ax.text(x=pd.to_datetime('2013-01-01'),y=6, s='Horizon', color='black',
       fontsize=16, fontweight='bold', alpha=0.8);
INFO:fbprophet:Disabling daily seasonality. Run prophet with daily_seasonality=True to override this.
INFO:fbprophet:Making 3 forecasts with cutoffs between 2013-01-20 00:00:00 and 2015-01-20 00:00:00
image.png

关于模拟的历史预测,可以看这个paper

The output of cross_validation is a dataframe with the true values y and the out-of-sample forecast values yhat, at each simulated forecast date and for each cutoff date. In particular, a forecast is made for every observed point between cutoff and cutoff + horizon. This dataframe can then be used to compute error measures of yhat vs. y.

Here we do cross-validation to assess prediction performance on a horizon of 365 days, starting with 730 days of training data in the first cutoff and then making predictions every 180 days. On this 8 year time series, this corresponds to 11 total forecasts.

可以使用函数cross_validation给这些历史截断点自动完成交叉验证
参数如下:

  1. horizon 从截断点往后预测多久
  2. initial 开始时间
  3. period 隔多久设置一个截断点
    默认情况下,periodhorizon的三倍,并且每隔半个horizon设置一个截断点

交叉验证的输出是一个DataFrame,包含真实的y和预测的yhat,可以用来评判效果

下面来个交叉验证,horizon=365天,initial=730天,period=180天,在八年的时间序列中,等于有11((365*8-730-365)/180)个总的预测

from fbprophet.diagnostics import cross_validation
df_cv = cross_validation(m, initial='730 days', period='180 days', horizon = '365 days')
df_cv.head()
INFO:fbprophet:Making 11 forecasts with cutoffs between 2010-02-15 00:00:00 and 2015-01-20 00:00:00
image.png

Python中,initial, period, horizon的字符串格式得是Pandas Timedelta,最大单位是天

函数performance_metrics可以用来评判模型效果,提供MSE,RMSE,MAE,MAPE,预估区间覆盖率等指标

from fbprophet.diagnostics import performance_metrics
df_p = performance_metrics(df_cv)
df_p.head()
image.png

可以使用plot_cross_validation_metric来可视化这些指标,下面看看mape的可视化,可以看到对未来一个月的预测有5%的误差,一年后的预测误差增加到了11%

from fbprophet.plot import plot_cross_validation_metric
fig = plot_cross_validation_metric(df_cv, metric='mape')
image.png

The size of the rolling window in the figure can be changed with the optional argument rolling_window, which specifies the proportion of forecasts to use in each rolling window. The default is 0.1, corresponding to 10% of rows from df_cv included in each window; increasing this will lead to a smoother average curve in the figure.

The initial period should be long enough to capture all of the components of the model, in particular seasonalities and extra regressors: at least a year for yearly seasonality, at least a week for weekly seasonality, etc.

可以使用参数rolling_window修改滚动窗口的大小,这个参数表示每个滚动窗口中预测的比例,默认0.1,相当于每个窗口中包含的df_cv的10%,增大这个值会让图中的平均曲线更光滑

初始周期应足够长,才能捕捉到各种东东,特别是季节性和额外回归因子,比如对于年季节性应该至少为一年(365 days),对于周季节性至少为一周(7 days)...

参考:
官方文档
trend_changepoints.ipynb
diagnostics.ipynb
时间序列模型Prophet使用详细讲解

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352