Time series forecast models can both make predictions and provide a confidence interval for those predictions.

Confidence intervals provide an upper and lower expectation for the real observation. These can be useful for assessing the range of real possible outcomes for a prediction and for better understanding the skill of the model

In this tutorial, you will discover how to calculate and interpret confidence intervals for time series forecasts with Python.

Specifically, you will learn:

- How to make a forecast with an ARIMA model and gather forecast diagnostic information.
- How to interpret a confidence interval for a forecast and configure different intervals.
- How to plot the confidence interval in the context of recent observations.

Let’s dive in.

## ARIMA Forecast

The ARIMA implementation in the statsmodels Python library can be used to fit an ARIMA model.

It returns an ARIMAResults object. This object provides the forecast() function that can be used to make predictions about future time steps and default to predicting the value at the next time step after the end of the training data.

Assuming we are predicting just the next time step, the forecast() method returns three values:

**Forecast**. The forecasted value in the units of the training time series.**Standard error**. The standard error for the model.**Confidence interval**. The 95% confidence interval for the forecast.

In this tutorial, we will better understand the confidence interval provided with an ARIMA forecast.

Before we dive in, let’s first look at the Daily Female Births dataset that we will use as the context for this tutorial.

### Stop learning Time Series Forecasting the *slow way*!

Take my free 7-day email course and discover how to get started (with sample code).

Click to sign-up and also get a free PDF Ebook version of the course.

## Daily Female Births Dataset

This dataset describes the number of daily female births in California in 1959.

The units are a count and there are 365 observations. The source of the dataset is credited to Newton (1988).

You can learn more and download the dataset from the Data Market website.

Download the dataset and save it in your current working directory with the filename “*daily-total-female-births.csv*“.

The example below loads and graphs the dataset.

1 2 3 4 5 |
from pandas import Series from matplotlib import pyplot series = Series.from_csv('daily-total-female-births.csv', header=0) series.plot() pyplot.show() |

Running the example loads the dataset and graphs it as a line plot.

## Forecast Confidence Interval

In this section, we will train an ARIMA model, use it to make a prediction, and inspect the confidence interval.

First, we will split the training dataset into a training and test dataset. Almost all observations will be used for training and we will hold back the last single observation as a test dataset for which we will make a prediction.

An ARIMA(5,1,1) model is trained. This is not the optimal model for this problem, just a good model for demonstration purposes.

The trained model is then used to make a prediction by calling the *forecast()* function. The results of the forecast are then printed.

The complete example is listed below.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
from pandas import Series from statsmodels.tsa.arima_model import ARIMA series = Series.from_csv('daily-total-female-births.csv', header=0) X = series.values X = X.astype('float32') size = len(X) - 1 train, test = X[0:size], X[size:] model = ARIMA(train, order=(5,1,1)) model_fit = model.fit(disp=False) forecast, stderr, conf = model_fit.forecast() print('Expected: %.3f' % test[0]) print('Forecast: %.3f' % forecast) print('Standard Error: %.3f' % stderr) print('95%% Confidence Interval: %.3f to %.3f' % (conf[0][0], conf[0][1])) |

Running the example prints the expected value from the test set followed by the predicted value, standard error, and confidence interval for the forecast.

1 2 3 4 |
Expected: 50.000 Forecast: 45.878 Standard Error: 6.996 95% Confidence Interval: 32.167 to 59.590 |

## Interpreting the Confidence Interval

The *forecast()* function allows the confidence interval to be specified.

The *alpha* argument on the *forecast()* function specifies the confidence level. It is set by default to alpha=0.05, which is a 95% confidence interval. This is a sensible and widely used confidence interval.

An alpha of 0.05 means that the ARIMA model will estimate the upper and lower values around the forecast where there is a only a 5% chance that the real value will not be in that range.

Put another way, the 95% confidence interval suggests that there is a high likelihood that the real observation will be within the range.

In the above example, the forecast was 45.878. The 95% confidence interval suggested that the real observation was highly likely to fall within the range of values between 32.167 and 59.590.

The real observation was 50.0 and was well within this range.

We can tighten the range of likely values a few ways:

- We can ask for a range that is narrower but increases the statistical likelihood of a real observation falling outside of the range.
- We can develop a model that has more predictive power and in turn makes more accurate predictions.

Further, the confidence interval is also limited by the assumptions made by the model, such as the distribution of errors made by the model fit a Gaussian distribution with a zero mean value (e.g. white noise).

Extending the example above, we can report our forecast with a few different commonly used confidence intervals of 80%, 90%, 95% and 99%.

The complete example is listed below.

1 2 3 4 5 6 7 8 9 10 11 12 13 |
from pandas import Series from statsmodels.tsa.arima_model import ARIMA series = Series.from_csv('daily-total-female-births.csv', header=0) X = series.values X = X.astype('float32') size = len(X) - 1 train, test = X[0:size], X[size:] model = ARIMA(train, order=(5,1,1)) model_fit = model.fit(disp=False) intervals = [0.2, 0.1, 0.05, 0.01] for a in intervals: forecast, stderr, conf = model_fit.forecast(alpha=a) print('%.1f%% Confidence Interval: %.3f between %.3f and %.3f' % ((1-a)*100, forecast, conf[0][0], conf[0][1])) |

Running the example prints the forecasts and confidence intervals for each *alpha* value.

We can see that we get the same forecast value each time and an interval that expands as our desire for a ‘safer’ interval increases. We can see that an 80% captures our actual value just fine in this specific case.

1 2 3 4 |
80.0% Confidence Interval: 45.878 between 36.913 and 54.844 90.0% Confidence Interval: 45.878 between 34.371 and 57.386 95.0% Confidence Interval: 45.878 between 32.167 and 59.590 99.0% Confidence Interval: 45.878 between 27.858 and 63.898 |

## Plotting the Confidence Interval

The confidence interval can be plotted directly.

The ARIMAResults object provides the plot_predict() function that can be used to make a forecast and plot the results showing recent observations, the forecast, and confidence interval.

As with the *forecast()* function, the confidence interval can be configured by specifying the *alpha* argument. The default is 0.05 (95% confidence), which is a sensible default.

The example below shows the same forecast from above plotted using this function.

1 2 3 4 5 6 7 8 9 10 11 12 |
from pandas import Series from matplotlib import pyplot from statsmodels.tsa.arima_model import ARIMA series = Series.from_csv('daily-total-female-births.csv', header=0) X = series.values X = X.astype('float32') size = len(X) - 1 train, test = X[0:size], X[size:] model = ARIMA(train, order=(5,1,1)) model_fit = model.fit(disp=False) model_fit.plot_predict(len(train)-10, len(train)+1) pyplot.show() |

The *plot_predict()* will plot the observed *y* values if the prediction interval covers the training data.

In this case, we predict the previous 10 days and the next 1 day. This is useful to see the prediction carry on from in sample to out of sample time indexes (blue). This is contracted with the actual observations from the last 10 days (green).

Finally, we can see the confidence interval as a gray cone around the predicted value. This is useful to get a spatial feeling for the range of possible values that an observation in the next time step may take.

## Summary

In this tutorial, you discovered how to calculate and interpret the confidence interval for a time series forecast with Python.

Specifically, you learned:

- How to report forecast diagnostic statistics when making a point forecast.
- How to interpret and configure the confidence interval for a time series forecast.
- How to plot a forecast and confidence interval in the context of recent observations.

Do you have any questions about forecast confidence intervals, or about this tutorial?

Ask your questions in the comments below and I will do my best to answer.

Hi, Jason.

your contents are great! Only one thing:

are you talking about “confidence intervals” or about “prediction intervals”? Some books mix the terms, but Time Series experts like Hyndman do an interesting differentiation:

https://robjhyndman.com/hyndsight/intervals/

Thanks

Prediction intervals. Thanks Luis!

Hi Jason.

What if we get a negative lower Confidence Interval while forecasting prices?? Since prices cannot be negative, is there a way to correct for this?

Thanks!

Great question. I’m not sure of good theory for this off the cuff. In practice, you can impose a hard limit on the interval to fit in your domain, for example:

Do you have additional code somewhere showing how to make predictions and plot the confidence interval for the entire time series using this model?

Thanks!

T

The post above does make a prediction and show the confidence interval.

You can use it as a template for your own dataset.

how can i get the confidence intervals for SARIMAX where i performed the grid search to get optimum parameters also keeping into consideration that i have a exogenous variable like Holiday effect?

Perhaps re-fit and analyze the best performing model as a standalone exercise after you have chosen its configuration.

I expect the approach in the above post would work for SARIMAX.

Great article. Thank you for all your posts. I learn a lot from them.

I understand that time series forecasting is for when we are forecasting the same variables in the future.

X1, X2, X3->X4, X5, X6

What if the nature of forecasting is not the same variables.

X1, X2, X3->Y1, Y2, Y3.

If we forecast one step, we will get something like:

X1, X2, X3, Y1-?>Y2, Y3

and we can not use the model further in this case for further predictions.

Do you know what is the technique used for such forecasting called?

Should we use a direct mapping between X->Y using LSTM or there is some other techniques available? I am looking for the definition of this problem and the techniques already developed. Thanks!

This is called sequence to sequence prediction:

https://machinelearningmastery.com/sequence-prediction/

Hi Jason,

Thank you very much for the article!

I have one question regarding to log transformation vs prediction intervals.

In case that we first log transformed our data to improve prediction power, when we’re getting back with our values to regular space we’re putting the log transformation on prediction intervals as well. But putting exp function on logged data for little differences in log space makes huge differences in regular space and making upper bounds gigantic in many cases.

So here is my question- are there any common methodologies how to avoid this side effect?

All best,

Mateusz

The prediction and interval assume a gaussian distribution that will balloon when made exponential was you comment.

Ouch. Good question.

I don’t know any good strategies off hand, sorry. Let me know if you discover anything.

Dear Jason: Thanks for the article. This is a great one. One question about the plot_predict call. As I read the code in “Plotting the Confidence Interval,” the test data is actually not used for ploting. Do you think if we should remove the test part from the code?

Sure.

Hi Jason,

Any idea, how to compute the confidence intervals using the LSTM model for timeseries forecasting ?

Thanks

Sanchit

You mean prediction intervals? Perhaps this will help:

https://machinelearningmastery.com/prediction-intervals-for-machine-learning/