시계열-퀵 가이드

시계열은 특정 기간 동안의 일련의 관찰입니다. 일 변량 시계열은 일정 기간 동안 주기적 시간 인스턴스에서 단일 변수가 취한 값으로 구성되며 다변량 시계열은 특정 기간 동안 동일한 주기적 시간 인스턴스에서 여러 변수가 취한 값으로 구성됩니다. 우리 모두가 매일 만나는 시계열의 가장 간단한 예는 일, 주, 월 또는 연도의 온도 변화입니다.

시간 데이터 분석은 변수가 시간에 따라 어떻게 변하는 지 또는 다른 변수 값의 변화에 ​​따라 어떻게 달라지는 지에 대한 유용한 통찰력을 제공 할 수 있습니다. 이전 값 및 / 또는 다른 변수에 대한 변수의 이러한 관계는 시계열 예측을 위해 분석 될 수 있으며 인공 지능에 수많은 응용 프로그램이 있습니다.

사용자가 기계 학습 문제를 다루거나 개발하려면 프로그래밍 언어에 대한 기본적인 이해가 필수적입니다. 머신 러닝 작업을 원하는 모든 사람에게 선호되는 프로그래밍 언어 목록은 다음과 같습니다.

파이썬

빠르고 쉽게 코딩 할 수있는 고급 해석 프로그래밍 언어입니다. Python은 절차 적 또는 객체 지향 프로그래밍 패러다임을 따를 수 있습니다. 다양한 라이브러리가 있기 때문에 복잡한 절차를 더 간단하게 구현할 수 있습니다. 이 튜토리얼에서는 Python으로 코딩 할 것이며 시계열 모델링에 유용한 해당 라이브러리는 다음 장에서 논의 될 것입니다.

아르 자형

Python과 마찬가지로 R은 통계 컴퓨팅 및 그래픽을 지원하는 해석 된 다중 패러다임 언어입니다. 다양한 패키지를 통해 R에서 기계 학습 모델링을 쉽게 구현할 수 있습니다.

자바

광범위한 패키지 가용성 및 정교한 데이터 시각화 기술로 널리 알려진 해석 된 객체 지향 프로그래밍 언어입니다.

C / C ++

이들은 컴파일 된 언어이며 가장 오래된 프로그래밍 언어 중 두 가지입니다. 이러한 언어는 ML 알고리즘의 구현을 쉽게 사용자 정의 할 수 있으므로 이미 기존 애플리케이션에 ML 기능을 통합하는 데 선호됩니다.

MATLAB

MATrix LABoratory는 매트릭스 작업 기능을 제공하는 다중 패러다임 언어입니다. 복잡한 문제에 대한 수학적 연산을 허용합니다. 주로 수치 연산에 사용되지만 일부 패키지는 그래픽 다중 도메인 시뮬레이션 및 모델 기반 설계도 허용합니다.

기계 학습 문제에 대해 선호되는 다른 프로그래밍 언어로는 JavaScript, LISP, Prolog, SQL, Scala, Julia, SAS 등이 있습니다.

Python은 작성하기 쉽고 이해하기 쉬운 코드 구조와 다양한 오픈 소스 라이브러리로 인해 기계 학습을 수행하는 개인들 사이에서 인기를 얻고 있습니다. 다음 장에서 사용할 오픈 소스 라이브러리 중 몇 가지가 아래에 소개되었습니다.

NumPy

Numerical Python은 과학 컴퓨팅에 사용되는 라이브러리입니다. N 차원 배열 객체에서 작동하며 크기, 모양, 평균, 표준 편차, 최소값, 최대 값과 같은 기본적인 수학적 기능과 선형 대수 함수 및 푸리에 변환과 같은 더 복잡한 함수를 제공합니다. 이 튜토리얼을 진행하면서 이에 대해 더 많이 배울 것입니다.

판다

이 라이브러리는 시리즈, 데이터 프레임 및 패널과 같은 매우 효율적이고 사용하기 쉬운 데이터 구조를 제공합니다. 단순한 데이터 수집 및 준비에서 데이터 분석에 이르기까지 Python의 기능이 향상되었습니다. Pandas와 NumPy라는 두 라이브러리는 소규모에서 매우 큰 데이터 세트에 대한 모든 작업을 매우 간단하게 만듭니다. 이러한 기능에 대해 자세히 알아 보려면이 튜토리얼을 따르십시오.

SciPy

Science Python은 과학 및 기술 컴퓨팅에 사용되는 라이브러리입니다. 최적화, 신호 및 이미지 처리, 통합, 보간 및 선형 대수를위한 기능을 제공합니다. 이 라이브러리는 기계 학습을 수행하는 동안 유용합니다. 이 튜토리얼을 진행하면서 이러한 기능에 대해 논의 할 것입니다.

Scikit 학습

이 라이브러리는 다양한 사용자 지정 가능한 회귀, 분류 및 클러스터링 모델을 포함하므로 통계 모델링, 기계 학습 및 딥 러닝에 널리 사용되는 SciPy Toolkit입니다. Numpy, Pandas 및 기타 라이브러리와 잘 작동하여 사용하기 쉽습니다.

Statsmodels

Scikit Learn과 마찬가지로이 라이브러리는 통계 데이터 탐색 및 통계 모델링에 사용됩니다. 또한 다른 Python 라이브러리와도 잘 작동합니다.

Matplotlib

이 라이브러리는 라인 플롯, 막대 그래프, 히트 맵, 산점도, 히스토그램 등과 같은 다양한 형식의 데이터 시각화에 사용됩니다. 플로팅에서 라벨링까지 필요한 모든 그래프 관련 기능이 포함되어 있습니다. 이 튜토리얼을 진행하면서 이러한 기능에 대해 논의 할 것입니다.

이러한 라이브러리는 모든 종류의 데이터로 기계 학습을 시작하는 데 매우 중요합니다.

위에서 논의한 것 외에도 시계열을 처리하는 데 특히 중요한 또 다른 라이브러리는 다음과 같습니다.

날짜 시간

이 라이브러리는 datetime과 calendar라는 두 개의 모듈로 시간 읽기, 형식화 및 조작에 필요한 모든 datetime 기능을 제공합니다.

우리는 다음 장에서이 라이브러리를 사용할 것입니다.

시계열은 등 간격 시간 간격으로 색인화 된 일련의 관측치입니다. 따라서 순서와 연속성은 모든 시계열에서 유지되어야합니다.

우리가 사용할 데이터 세트는 상당히 오염 된 이탈리아 도시의 대기 질에 대해 약 1 년 동안의 시간별 데이터를 포함하는 다 변수 시계열입니다. 데이터 세트는 아래 링크에서 다운로드 할 수 있습니다.https://archive.ics.uci.edu/ml/datasets/air+quality.

다음을 확인해야합니다.

  • 시계열의 간격이 동일하고

  • 중복 된 값이나 간격이 없습니다.

시계열이 연속적이지 않은 경우 업 샘플링 또는 다운 샘플링 할 수 있습니다.

df.head () 표시

[122]에서 :

import pandas

[123]에서 :

df = pandas.read_csv("AirQualityUCI.csv", sep = ";", decimal = ",")
df = df.iloc[ : , 0:14]

[124] :

len(df)

출력 [124] :

9471

[125] :

df.head()

출력 [125] :

시계열을 전처리하기 위해 데이터 세트에 NaN (NULL) 값이 없는지 확인합니다. 있는 경우 0 또는 평균 또는 선행 또는 후속 값으로 바꿀 수 있습니다. 교체는 시계열의 연속성이 유지되도록 드롭보다 선호되는 선택입니다. 그러나 데이터 세트에서 마지막 몇 개의 값은 NULL 인 것처럼 보이므로 삭제해도 연속성에 영향을주지 않습니다.

NaN (Not-a-Number) 삭제

[126] :

df.isna().sum()
Out[126]:
Date             114
Time             114
CO(GT)           114
PT08.S1(CO)      114
NMHC(GT)         114
C6H6(GT)         114
PT08.S2(NMHC)    114
NOx(GT)          114
PT08.S3(NOx)     114
NO2(GT)          114
PT08.S4(NO2)     114
PT08.S5(O3)      114
T                114
RH               114
dtype: int64

[127] :

df = df[df['Date'].notnull()]

[128] :

df.isna().sum()

출력 [128] :

Date             0
Time             0
CO(GT)           0
PT08.S1(CO)      0
NMHC(GT)         0
C6H6(GT)         0
PT08.S2(NMHC)    0
NOx(GT)          0
PT08.S3(NOx)     0
NO2(GT)          0
PT08.S4(NO2)     0
PT08.S5(O3)      0
T                0
RH               0
dtype: int64

시계열은 일반적으로 시간에 대한 선 그래프로 표시됩니다. 이를 위해 이제 날짜 및 시간 열을 결합하여 문자열에서 datetime 객체로 변환합니다. 이는 datetime 라이브러리를 사용하여 수행 할 수 있습니다.

datetime 객체로 변환

[129] :

df['DateTime'] = (df.Date) + ' ' + (df.Time)
print (type(df.DateTime[0]))

<class 'str'>

[130] :

import datetime

df.DateTime = df.DateTime.apply(lambda x: datetime.datetime.strptime(x, '%d/%m/%Y %H.%M.%S'))
print (type(df.DateTime[0]))

<class 'pandas._libs.tslibs.timestamps.Timestamp'>

온도와 같은 일부 변수가 시간의 변화에 ​​따라 어떻게 변하는 지 살펴 보겠습니다.

플롯 표시

[131]에서 :

df.index = df.DateTime

[132]에서 :

import matplotlib.pyplot as plt
plt.plot(df['T'])

출력 [132] :

[<matplotlib.lines.Line2D at 0x1eaad67f780>]

[208]에서 :

plt.plot(df['C6H6(GT)'])

출력 [208] :

[<matplotlib.lines.Line2D at 0x1eaaeedff28>]

상자 그림은 데이터 세트에 대한 많은 정보를 단일 그래프로 압축 할 수있는 또 다른 유용한 유형의 그래프입니다. 하나 또는 여러 변수의 평균, 25 % 및 75 % 사 분위수 및 특이 치를 보여줍니다. 특이 치의 수가 적고 평균에서 매우 멀리 떨어져있는 경우 특이 치를 평균값 또는 75 % 사 분위수 값으로 설정하여 특이 치를 제거 할 수 있습니다.

상자 그림 표시

[134] :

plt.boxplot(df[['T','C6H6(GT)']].values)

출력 [134] :

{'whiskers': [<matplotlib.lines.Line2D at 0x1eaac16de80>,
   <matplotlib.lines.Line2D at 0x1eaac16d908>,
   <matplotlib.lines.Line2D at 0x1eaac177a58>,
   <matplotlib.lines.Line2D at 0x1eaac177cf8>],
   'caps': [<matplotlib.lines.Line2D at 0x1eaac16d2b0>,
   <matplotlib.lines.Line2D at 0x1eaac16d588>,
   <matplotlib.lines.Line2D at 0x1eaac1a69e8>,
   <matplotlib.lines.Line2D at 0x1eaac1a64a8>],
   'boxes': [<matplotlib.lines.Line2D at 0x1eaac16dc50>,
   <matplotlib.lines.Line2D at 0x1eaac1779b0>],
   'medians': [<matplotlib.lines.Line2D at 0x1eaac16d4a8>,
   <matplotlib.lines.Line2D at 0x1eaac1a6c50>],
   'fliers': [<matplotlib.lines.Line2D at 0x1eaac177dd8>,
   <matplotlib.lines.Line2D at 0x1eaac1a6c18>],'means': []
}

소개

시계열에는 아래와 같이 4 개의 성분이 있습니다.

  • Level − 계열이 변하는 평균값입니다.

  • Trend − 시간에 따른 변수의 증가 또는 감소 동작입니다.

  • Seasonality − 시계열의 주기적 동작입니다.

  • Noise − 환경 적 요인으로 추가 된 관측치의 오차입니다.

시계열 모델링 기법

이러한 구성 요소를 캡처하기 위해 널리 사용되는 시계열 모델링 기술이 많이 있습니다. 이 섹션에서는 각 기술에 대해 간략하게 소개하지만 다음 장에서 자세히 설명합니다.

나이브 방법

이것은 예측 값이 시간 종속 변수의 이전 값 또는 이전 실제 값의 평균과 같은 값이 주어지는 것과 같은 간단한 추정 기법입니다. 이들은 정교한 모델링 기술과의 비교에 사용됩니다.

자동 회귀

자동 회귀는 이전 기간 값의 함수로 미래 기간 값을 예측합니다. 자동 회귀 예측은 순진한 방법보다 데이터에 더 적합 할 수 있지만 계절성을 고려하지 못할 수 있습니다.

ARIMA 모델

자동 회귀 통합 이동 평균은 고정 시계열의 이전 시간 단계에서 이전 값 및 잔여 오류의 선형 함수로 변수 값을 모델링합니다. 그러나 실제 데이터는 고정적이지 않고 계절성이있을 수 있으므로 Seasonal-ARIMA 및 Fractional-ARIMA가 개발되었습니다. ARIMA는 일 변량 시계열에서 작동하여 여러 변수를 처리하기 위해 VARIMA가 도입되었습니다.

지수 평활

변수의 값을 이전 값의 지수 가중 선형 함수로 모델링합니다. 이 통계 모델은 추세와 계절성도 처리 할 수 ​​있습니다.

LSTM

장단기 기억 모델 (LSTM)은 장기 종속성을 설명하기 위해 시계열에 사용되는 순환 신경망입니다. 다 변수 시계열의 추세를 포착하기 위해 많은 양의 데이터로 훈련 할 수 있습니다.

상기 모델링 기술은 시계열 회귀에 사용됩니다. 다음 장에서 이제이 모든 것을 하나씩 살펴 보겠습니다.

소개

모든 통계 또는 기계 학습 모델에는 데이터 모델링 방법에 큰 영향을 미치는 몇 가지 매개 변수가 있습니다. 예를 들어 ARIMA에는 p, d, q 값이 있습니다. 이러한 매개 변수는 실제 값과 모델링 된 값 사이의 오차가 최소가되도록 결정되어야합니다. 매개 변수 교정은 모델 피팅에서 가장 중요하고 시간이 많이 걸리는 작업이라고합니다. 따라서 최적의 매개 변수를 선택하는 것은 매우 중요합니다.

매개 변수 교정 방법

매개 변수를 보정하는 방법에는 여러 가지가 있습니다. 이 섹션에서는 이들 중 일부에 대해 자세히 설명합니다.

뺑소니

모델을 보정하는 일반적인 방법 중 하나는 시계열을 시각화하는 것으로 시작하여 몇 가지 매개 변수 값을 직관적으로 시도하고 충분히 적합 할 때까지 계속해서 변경하는 수동 보정입니다. 우리가 시도하는 모델을 잘 이해해야합니다. ARIMA 모델의 경우 'p'매개 변수에 대한 자동 상관 플롯, 'q'매개 변수에 대한 부분 자기 상관 플롯 및 ADF 테스트를 통해 시계열의 정상 성을 확인하고 'd'매개 변수를 설정하여 손 보정을 수행합니다. . 이 모든 내용은 다음 장에서 자세히 설명합니다.

그리드 검색

모델을 보정하는 또 다른 방법은 그리드 검색을 사용하는 것입니다. 이는 기본적으로 가능한 모든 매개 변수 조합에 대해 모델을 구축하고 오류가 최소 인 모델을 선택하는 것을 의미합니다. 이는 시간이 많이 걸리므로 교정 할 매개 변수의 수가 많고 여러 중첩 for 루프를 포함하므로 값의 범위가 더 적을 때 유용합니다.

유전 알고리즘

유전 알고리즘은 좋은 솔루션이 결국 가장 '최적'솔루션으로 진화한다는 생물학적 원리에 따라 작동합니다. 그것은 궁극적으로 최적의 솔루션에 도달하기 위해 돌연변이, 교차 및 선택의 생물학적 작업을 사용합니다.

더 많은 지식을 얻으려면 Bayesian 최적화 및 Swarm 최적화와 같은 다른 매개 변수 최적화 기술에 대해 읽을 수 있습니다.

소개

시간 't'에서 예측 된 값을 시간 't-1'에서 변수의 실제 값 또는 시리즈의 롤링 평균으로 가정하는 것과 같은 순진한 방법을 사용하여 통계 모델 및 기계 학습 모델이 얼마나 잘 수행 할 수 있는지 평가합니다. 그들의 필요를 강조합니다.

이 장에서는 시계열 데이터의 기능 중 하나에 대해 이러한 모델을 시도해 보겠습니다.

먼저 데이터의 '온도'기능의 평균과 그 주변의 편차를 살펴 보겠습니다. 최대 및 최소 온도 값을 확인하는 것도 유용합니다. 여기서 numpy 라이브러리의 기능을 사용할 수 있습니다.

통계 표시

[135] :

import numpy
print (
   'Mean: ',numpy.mean(df['T']), '; 
   Standard Deviation: ',numpy.std(df['T']),'; 
   \nMaximum Temperature: ',max(df['T']),'; 
   Minimum Temperature: ',min(df['T'])
)

데이터를 이해하는 데 유용한 등 간격 타임 라인에서 9357 개의 모든 관측치에 대한 통계가 있습니다.

이제 첫 번째 순진한 방법을 시도하여 현재 예측 값을 이전 시간의 실제 값과 동일하게 설정하고이 방법의 성능을 정량화하기 위해 평균 제곱근 오차 (RMSE)를 계산합니다.

1 개 표시 세인트 나이브 방법

[136] :

df['T']
df['T_t-1'] = df['T'].shift(1)

[137]에서 :

df_naive = df[['T','T_t-1']][1:]

[138]에서 :

from sklearn import metrics
from math import sqrt

true = df_naive['T']
prediction = df_naive['T_t-1']
error = sqrt(metrics.mean_squared_error(true,prediction))
print ('RMSE for Naive Method 1: ', error)

Naive 방법 1에 대한 RMSE : 12.901140576492974

현재 예측 된 값이 이전 기간의 평균과 동일한 다음 순진한 방법을 살펴 보겠습니다. 이 방법에 대한 RMSE도 계산합니다.

번째 순진한 방법 표시

[139]에서 :

df['T_rm'] = df['T'].rolling(3).mean().shift(1)
df_naive = df[['T','T_rm']].dropna()

[140] :

true = df_naive['T']
prediction = df_naive['T_rm']
error = sqrt(metrics.mean_squared_error(true,prediction))
print ('RMSE for Naive Method 2: ', error)

RMSE for Naive Method 2: 14.957633272839242

여기에서 고려하려는 '지연'이라고도하는 다양한 이전 기간을 실험 할 수 있으며 여기에서는 3으로 유지됩니다. 이 데이터에서 지연 수를 늘리면 오류가 증가 함을 알 수 있습니다. 지연이 1로 유지되면 이전에 사용한 순진한 방법과 동일하게됩니다.

Points to Note

  • 제곱 평균 오차를 계산하는 매우 간단한 함수를 작성할 수 있습니다. 여기서는 'sklearn'패키지의 평균 제곱 오차 함수를 사용한 다음 제곱근을 취했습니다.

  • pandas에서 df [ 'column_name']은 df.column_name으로도 쓸 수 있지만,이 데이터 세트의 경우 df.T는 df [ 'T']와 동일하게 작동하지 않습니다. df.T는 데이터 프레임을 전치하는 함수이기 때문입니다. 따라서 df [ 'T'] 만 사용하거나 다른 구문을 사용하기 전에이 열의 이름을 바꾸는 것을 고려하십시오.

고정 시계열의 경우 자동 회귀 모델은 시간 't'의 변수 값을 앞에있는 값 'p'시간 단계의 선형 함수로 봅니다. 수학적으로 다음과 같이 쓸 수 있습니다.

$$y_{t} = \:C+\:\phi_{1}y_{t-1}\:+\:\phi_{2}Y_{t-2}+...+\phi_{p}y_{t-p}+\epsilon_{t}$$

 

여기서 'p'는 자동 회귀 추세 매개 변수입니다.

$\epsilon_{t}$ 백색 잡음이고

$y_{t-1}, y_{t-2}\:\: ...y_{t-p}$ 이전 기간의 변수 값을 나타냅니다.

p의 값은 다양한 방법을 사용하여 보정 할 수 있습니다. 'p'의 적절한 값을 찾는 한 가지 방법은 자동 상관 플롯을 그리는 것입니다.

Note− 테스트 데이터는 우리 모델의 정확성을 알아 내기위한 것이며 우리가 사용할 수 없다고 가정하기 때문에 데이터에 대한 분석을 수행하기 전에 사용 가능한 총 데이터의 8 : 2 비율로 데이터를 훈련 및 테스트해야합니다. 예측이 이루어질 때까지. 시계열의 경우 데이터 포인트의 순서가 매우 중요하므로 데이터 분할 중에 순서를 잃지 않도록 유의해야합니다.

자동 상관 플롯 또는 상관 관계는 이전 시간 단계에서 변수 자체와의 관계를 보여줍니다. Pearson의 상관 관계를 사용하고 95 % 신뢰 구간 내에서 상관 관계를 보여줍니다. 데이터의 '온도'변수가 어떻게 보이는지 살펴 보겠습니다.

ACP 표시

[141]에서 :

split = len(df) - int(0.2*len(df))
train, test = df['T'][0:split], df['T'][split:]

[142]에서 :

from statsmodels.graphics.tsaplots import plot_acf

plot_acf(train, lags = 100)
plt.show()

음영 처리 된 파란색 영역 밖에있는 모든 지연 값은 csorrelation을 갖는 것으로 간주됩니다.

고정 시계열의 경우 이동 평균 모델은 시간 't'의 변수 값을 앞의 'q'시간 단계에서 발생한 잔차 오류의 선형 함수로 봅니다. 잔차 오차는 't'시점의 값을 이전 값의 이동 평균과 비교하여 계산됩니다.

수학적으로 다음과 같이 쓸 수 있습니다.

$$y_{t} = c\:+\:\epsilon_{t}\:+\:\theta_{1}\:\epsilon_{t-1}\:+\:\theta_{2}\:\epsilon_{t-2}\:+\:...+:\theta_{q}\:\epsilon_{t-q}\:$$

여기서 'q'는 이동 평균 추세 매개 변수입니다.

$\epsilon_{t}$ 백색 잡음이고

$\epsilon_{t-1}, \epsilon_{t-2}...\epsilon_{t-q}$ 이전 기간의 오류 용어입니다.

'q'의 값은 다양한 방법으로 보정 할 수 있습니다. 'q'의 적절한 값을 찾는 한 가지 방법은 부분 자기 상관 플롯을 그리는 것입니다.

부분 자기 상관 플롯은 간접 상관 관계가 제거 된 이전 시간 단계에서 변수 자체와의 관계를 보여줍니다. 직접 상관 관계와 간접 상관 관계를 보여주는 자기 상관 플롯과는 달리 우리의 '온도'변수가 어떻게 보이는지 살펴 보겠습니다. 데이터.

PACP 표시

[143]에서 :

from statsmodels.graphics.tsaplots import plot_pacf

plot_pacf(train, lags = 100)
plt.show()

부분 자기 상관은 상관 관계 도와 동일한 방식으로 읽습니다.

고정 시계열의 경우 시간 't'의 변수가 이전 관측치 또는 잔차 오류의 선형 함수라는 것을 이미 이해했습니다. 따라서 두 가지를 결합하고 자동 회귀 이동 평균 (ARMA) 모델을 사용할 때입니다.

그러나 때때로 시계열은 고정되어 있지 않습니다. 즉, 평균과 같은 시리즈의 통계적 속성은 시간이 지남에 따라 변합니다. 그리고 지금까지 연구 한 통계 모델은 시계열이 고정 된 것으로 가정하므로 시계열을 고정시켜 시계열을 차등화하는 전처리 단계를 포함 할 수 있습니다. 이제 우리가 다루고있는 시계열이 고정되어 있는지 여부를 알아내는 것이 중요합니다.

시계열의 정상 성을 찾는 다양한 방법은 시계열 플롯에서 계절성 또는 추세를 찾고, 다양한 기간에 대한 평균과 분산의 차이를 확인하고, ADF (Augmented Dickey-Fuller) 테스트, KPSS 테스트, Hurst 지수 등 .

데이터 세트의 '온도'변수가 고정 시계열인지 ADF 테스트를 사용하지 않는지 살펴 보겠습니다.

[74] :

from statsmodels.tsa.stattools import adfuller

result = adfuller(train)
print('ADF Statistic: %f' % result[0])
print('p-value: %f' % result[1])
print('Critical Values:')
for key, value In result[4].items()
   print('\t%s: %.3f' % (key, value))

ADF 통계 : -10.406056

p- 값 : 0.000000

중요한 값 :

1 % : -3.431

5 % : -2.862

10 % : -2.567

이제 ADF 테스트를 실행 했으므로 결과를 해석해 보겠습니다. 먼저 ADF 통계를 임계 값과 비교합니다. 임계 값이 낮을수록 시리즈가 비정상 일 가능성이 가장 높다는 것을 알 수 있습니다. 다음으로, 우리는 p- 값을 봅니다. 0.05보다 큰 p- 값은 시계열이 비정상임을 나타냅니다.

또는 0.05보다 작거나 같은 p- 값 또는 임계 값보다 작은 ADF 통계는 시계열이 정상적임을 나타냅니다.

따라서 우리가 다루는 시계열은 이미 고정되어 있습니다. 고정 시계열의 경우 'd'매개 변수를 0으로 설정합니다.

Hurst 지수를 사용하여 시계열의 정상 성을 확인할 수도 있습니다.

[75] :

import hurst

H, c,data = hurst.compute_Hc(train)
print("H = {:.4f}, c = {:.4f}".format(H,c))

H = 0.1660, c = 5.0740

H <0.5의 값은 반 지속적 동작을 나타내고 H> 0.5는 지속적 동작 또는 추세 계열을 나타냅니다. H = 0.5는 무작위 걷기 / 브라운 운동을 보여줍니다. H <0.5의 값은 시리즈가 고정되어 있음을 확인합니다.

For non-stationary time series, we set ‘d’ parameter as 1. Also, the value of the auto-regressive trend parameter ‘p’ and the moving average trend parameter ‘q’, is calculated on the stationary time series i.e by plotting ACP and PACP after differencing the time series.

ARIMA Model, which is characterized by 3 parameter, (p,d,q) are now clear to us, so let us model our time series and predict the future values of temperature.

In [156]:

from statsmodels.tsa.arima_model import ARIMA

model = ARIMA(train.values, order=(5, 0, 2))
model_fit = model.fit(disp=False)

In [157]:

predictions = model_fit.predict(len(test))
test_ = pandas.DataFrame(test)
test_['predictions'] = predictions[0:1871]

In [158]:

plt.plot(df['T'])
plt.plot(test_.predictions)
plt.show()

In [167]:

error = sqrt(metrics.mean_squared_error(test.values,predictions[0:1871]))
print ('Test RMSE for ARIMA: ', error)

Test RMSE for ARIMA: 43.21252940234892

In the previous chapter, we have now seen how ARIMA model works, and its limitations that it cannot handle seasonal data or multivariate time series and hence, new models were introduced to include these features.

A glimpse of these new models is given here −

Vector Auto-Regression (VAR)

It is a generalized version of auto regression model for multivariate stationary time series. It is characterized by ‘p’ parameter.

Vector Moving Average (VMA)

It is a generalized version of moving average model for multivariate stationary time series. It is characterized by ‘q’ parameter.

Vector Auto Regression Moving Average (VARMA)

It is the combination of VAR and VMA and a generalized version of ARMA model for multivariate stationary time series. It is characterized by ‘p’ and ‘q’ parameters. Much like, ARMA is capable of acting like an AR model by setting ‘q’ parameter as 0 and as a MA model by setting ‘p’ parameter as 0, VARMA is also capable of acting like an VAR model by setting ‘q’ parameter as 0 and as a VMA model by setting ‘p’ parameter as 0.

In [209]:

df_multi = df[['T', 'C6H6(GT)']]
split = len(df) - int(0.2*len(df))
train_multi, test_multi = df_multi[0:split], df_multi[split:]

In [211]:

from statsmodels.tsa.statespace.varmax import VARMAX

model = VARMAX(train_multi, order = (2,1))
model_fit = model.fit()
c:\users\naveksha\appdata\local\programs\python\python37\lib\site-packages\statsmodels\tsa\statespace\varmax.py:152: 
   EstimationWarning: Estimation of VARMA(p,q) models is not generically robust, 
   due especially to identification issues. 
   EstimationWarning)
c:\users\naveksha\appdata\local\programs\python\python37\lib\site-packages\statsmodels\tsa\base\tsa_model.py:171: 
   ValueWarning: No frequency information was provided, so inferred frequency H will be used. 
  % freq, ValueWarning)
c:\users\naveksha\appdata\local\programs\python\python37\lib\site-packages\statsmodels\base\model.py:508: 
   ConvergenceWarning: Maximum Likelihood optimization failed to converge. Check mle_retvals 
  "Check mle_retvals", ConvergenceWarning)

In [213]:

predictions_multi = model_fit.forecast( steps=len(test_multi))
c:\users\naveksha\appdata\local\programs\python\python37\lib\site-packages\statsmodels\tsa\base\tsa_model.py:320: 
   FutureWarning: Creating a DatetimeIndex by passing range endpoints is deprecated.  Use `pandas.date_range` instead.
   freq = base_index.freq)
c:\users\naveksha\appdata\local\programs\python\python37\lib\site-packages\statsmodels\tsa\statespace\varmax.py:152: 
   EstimationWarning: Estimation of VARMA(p,q) models is not generically robust, due especially to identification issues.
   EstimationWarning)

In [231]:

plt.plot(train_multi['T'])
plt.plot(test_multi['T'])
plt.plot(predictions_multi.iloc[:,0:1], '--')
plt.show()

plt.plot(train_multi['C6H6(GT)'])
plt.plot(test_multi['C6H6(GT)'])
plt.plot(predictions_multi.iloc[:,1:2], '--')
plt.show()

The above code shows how VARMA model can be used to model multivariate time series, although this model may not be best suited on our data.

VARMA with Exogenous Variables (VARMAX)

It is an extension of VARMA model where extra variables called covariates are used to model the primary variable we are interested it.

Seasonal Auto Regressive Integrated Moving Average (SARIMA)

This is the extension of ARIMA model to deal with seasonal data. It divides the data into seasonal and non-seasonal components and models them in a similar fashion. It is characterized by 7 parameters, for non-seasonal part (p,d,q) parameters same as for ARIMA model and for seasonal part (P,D,Q,m) parameters where ‘m’ is the number of seasonal periods and P,D,Q are similar to parameters of ARIMA model. These parameters can be calibrated using grid search or genetic algorithm.

SARIMA with Exogenous Variables (SARIMAX)

This is the extension of SARIMA model to include exogenous variables which help us to model the variable we are interested in.

It may be useful to do a co-relation analysis on variables before putting them as exogenous variables.

In [251]:

from scipy.stats.stats import pearsonr
x = train_multi['T'].values
y = train_multi['C6H6(GT)'].values

corr , p = pearsonr(x,y)
print ('Corelation Coefficient =', corr,'\nP-Value =',p)
Corelation Coefficient = 0.9701173437269858
P-Value = 0.0

Pearson’s Correlation shows a linear relation between 2 variables, to interpret the results, we first look at the p-value, if it is less that 0.05 then the value of coefficient is significant, else the value of coefficient is not significant. For significant p-value, a positive value of correlation coefficient indicates positive correlation, and a negative value indicates a negative correlation.

Hence, for our data, ‘temperature’ and ‘C6H6’ seem to have a highly positive correlation. Therefore, we will

In [297]:

from statsmodels.tsa.statespace.sarimax import SARIMAX

model = SARIMAX(x, exog = y, order = (2, 0, 2), seasonal_order = (2, 0, 1, 1), enforce_stationarity=False, enforce_invertibility = False)
model_fit = model.fit(disp = False)
c:\users\naveksha\appdata\local\programs\python\python37\lib\site-packages\statsmodels\base\model.py:508: 
   ConvergenceWarning: Maximum Likelihood optimization failed to converge. Check mle_retvals
   "Check mle_retvals", ConvergenceWarning)

In [298]:

y_ = test_multi['C6H6(GT)'].values
predicted = model_fit.predict(exog=y_)
test_multi_ = pandas.DataFrame(test)
test_multi_['predictions'] = predicted[0:1871]

In [299]:

plt.plot(train_multi['T'])
plt.plot(test_multi_['T'])
plt.plot(test_multi_.predictions, '--')

Out[299]:

[<matplotlib.lines.Line2D at 0x1eab0191c18>]

The predictions here seem to take larger variations now as opposed to univariate ARIMA modelling.

Needless to say, SARIMAX can be used as an ARX, MAX, ARMAX or ARIMAX model by setting only the corresponding parameters to non-zero values.

Fractional Auto Regressive Integrated Moving Average (FARIMA)

At times, it may happen that our series is not stationary, yet differencing with ‘d’ parameter taking the value 1 may over-difference it. So, we need to difference the time series using a fractional value.

In the world of data science there is no one superior model, the model that works on your data depends greatly on your dataset. Knowledge of various models allows us to choose one that work on our data and experimenting with that model to achieve the best results. And results should be seen as plot as well as error metrics, at times a small error may also be bad, hence, plotting and visualizing the results is essential.

In the next chapter, we will be looking at another statistical model, exponential smoothing.

In this chapter, we will talk about the techniques involved in exponential smoothing of time series.

Simple Exponential Smoothing

Exponential Smoothing is a technique for smoothing univariate time-series by assigning exponentially decreasing weights to data over a time period.

Mathematically, the value of variable at time ‘t+1’ given value at time t, y_(t+1|t) is defined as −

$$y_{t+1|t}\:=\:\alpha y_{t}\:+\:\alpha\lgroup1 -\alpha\rgroup y_{t-1}\:+\alpha\lgroup1-\alpha\rgroup^{2}\:y_{t-2}\:+\:...+y_{1}$$

where,$0\leq\alpha \leq1$ is the smoothing parameter, and

$y_{1},....,y_{t}$ are previous values of network traffic at times 1, 2, 3, … ,t.

This is a simple method to model a time series with no clear trend or seasonality. But exponential smoothing can also be used for time series with trend and seasonality.

Triple Exponential Smoothing

Triple Exponential Smoothing (TES) or Holt's Winter method, applies exponential smoothing three times - level smoothing $l_{t}$, trend smoothing $b_{t}$, and seasonal smoothing $S_{t}$, with $\alpha$, $\beta^{*}$ and $\gamma$ as smoothing parameters with ‘m’ as the frequency of the seasonality, i.e. the number of seasons in a year.

According to the nature of the seasonal component, TES has two categories −

  • Holt-Winter's Additive Method − When the seasonality is additive in nature.

  • Holt-Winter’s Multiplicative Method − When the seasonality is multiplicative in nature.

For non-seasonal time series, we only have trend smoothing and level smoothing, which is called Holt’s Linear Trend Method.

Let’s try applying triple exponential smoothing on our data.

In [316]:

from statsmodels.tsa.holtwinters import ExponentialSmoothing

model = ExponentialSmoothing(train.values, trend= )
model_fit = model.fit()

In [322]:

predictions_ = model_fit.predict(len(test))

In [325]:

plt.plot(test.values)
plt.plot(predictions_[1:1871])

Out[325]:

[<matplotlib.lines.Line2D at 0x1eab00f1cf8>]

Here, we have trained the model once with training set and then we keep on making predictions. A more realistic approach is to re-train the model after one or more time step(s). As we get the prediction for time ‘t+1’ from training data ‘til time ‘t’, the next prediction for time ‘t+2’ can be made using the training data ‘til time ‘t+1’ as the actual value at ‘t+1’ will be known then. This methodology of making predictions for one or more future steps and then re-training the model is called rolling forecast or walk forward validation.

In time series modelling, the predictions over time become less and less accurate and hence it is a more realistic approach to re-train the model with actual data as it gets available for further predictions. Since training of statistical models are not time consuming, walk-forward validation is the most preferred solution to get most accurate results.

Let us apply one step walk forward validation on our data and compare it with the results we got earlier.

In [333]:

prediction = []
data = train.values
for t In test.values:
   model = (ExponentialSmoothing(data).fit())
   y = model.predict()
   prediction.append(y[0])
   data = numpy.append(data, t)

In [335]:

test_ = pandas.DataFrame(test)
test_['predictionswf'] = prediction

In [341]:

plt.plot(test_['T'])
plt.plot(test_.predictionswf, '--')
plt.show()

In [340]:

error = sqrt(metrics.mean_squared_error(test.values,prediction))
print ('Test RMSE for Triple Exponential Smoothing with Walk-Forward Validation: ', error)
Test RMSE for Triple Exponential Smoothing with Walk-Forward Validation:  11.787532205759442

We can see that our model performs significantly better now. In fact, the trend is followed so closely that on the plot predictions are overlapping with the actual values. You can try applying walk-forward validation on ARIMA models too.

In 2017, Facebook open sourced the prophet model which was capable of modelling the time series with strong multiple seasonalities at day level, week level, year level etc. and trend. It has intuitive parameters that a not-so-expert data scientist can tune for better forecasts. At its core, it is an additive regressive model which can detect change points to model the time series.

Prophet decomposes the time series into components of trend $g_{t}$, seasonality $S_{t}$ and holidays $h_{t}$.

$$y_{t}=g_{t}+s_{t}+h_{t}+\epsilon_{t}$$

Where, $\epsilon_{t}$ is the error term.

Similar packages for time series forecasting such as causal impact and anomaly detection were introduced in R by google and twitter respectively.

Now, we are familiar with statistical modelling on time series, but machine learning is all the rage right now, so it is essential to be familiar with some machine learning models as well. We shall start with the most popular model in time series domain − Long Short-term Memory model.

LSTM is a class of recurrent neural network. So before we can jump to LSTM, it is essential to understand neural networks and recurrent neural networks.

Neural Networks

An artificial neural network is a layered structure of connected neurons, inspired by biological neural networks. It is not one algorithm but combinations of various algorithms which allows us to do complex operations on data.

Recurrent Neural Networks

It is a class of neural networks tailored to deal with temporal data. The neurons of RNN have a cell state/memory, and input is processed according to this internal state, which is achieved with the help of loops with in the neural network. There are recurring module(s) of ‘tanh’ layers in RNNs that allow them to retain information. However, not for a long time, which is why we need LSTM models.

LSTM

It is special kind of recurrent neural network that is capable of learning long term dependencies in data. This is achieved because the recurring module of the model has a combination of four layers interacting with each other.

The picture above depicts four neural network layers in yellow boxes, point wise operators in green circles, input in yellow circles and cell state in blue circles. An LSTM module has a cell state and three gates which provides them with the power to selectively learn, unlearn or retain information from each of the units. The cell state in LSTM helps the information to flow through the units without being altered by allowing only a few linear interactions. Each unit has an input, output and a forget gate which can add or remove the information to the cell state. The forget gate decides which information from the previous cell state should be forgotten for which it uses a sigmoid function. The input gate controls the information flow to the current cell state using a point-wise multiplication operation of ‘sigmoid’ and ‘tanh’ respectively. Finally, the output gate decides which information should be passed on to the next hidden state

Now that we have understood the internal working of LSTM model, let us implement it. To understand the implementation of LSTM, we will start with a simple example − a straight line. Let us see, if LSTM can learn the relationship of a straight line and predict it.

First let us create the dataset depicting a straight line.

In [402]:

x = numpy.arange (1,500,1)
y = 0.4 * x + 30
plt.plot(x,y)

Out[402]:

[<matplotlib.lines.Line2D at 0x1eab9d3ee10>]

In [403]:

trainx, testx = x[0:int(0.8*(len(x)))], x[int(0.8*(len(x))):]
trainy, testy = y[0:int(0.8*(len(y)))], y[int(0.8*(len(y))):]
train = numpy.array(list(zip(trainx,trainy)))
test = numpy.array(list(zip(trainx,trainy)))

Now that the data has been created and split into train and test. Let’s convert the time series data into the form of supervised learning data according to the value of look-back period, which is essentially the number of lags which are seen to predict the value at time ‘t’.

So a time series like this −

time variable_x
t1  x1
t2  x2
 :   :
 :   :
T   xT

When look-back period is 1, is converted to −

x1   x2
x2   x3
 :    :
 :    :
xT-1 xT

In [404]:

def create_dataset(n_X, look_back):
   dataX, dataY = [], []
   for i in range(len(n_X)-look_back):
      a = n_X[i:(i+look_back), ]
      dataX.append(a)
      dataY.append(n_X[i + look_back, ])
   return numpy.array(dataX), numpy.array(dataY)

In [405]:

look_back = 1
trainx,trainy = create_dataset(train, look_back)
testx,testy = create_dataset(test, look_back)

trainx = numpy.reshape(trainx, (trainx.shape[0], 1, 2))
testx = numpy.reshape(testx, (testx.shape[0], 1, 2))

Now we will train our model.

Small batches of training data are shown to network, one run of when entire training data is shown to the model in batches and error is calculated is called an epoch. The epochs are to be run ‘til the time the error is reducing.

In [ ]:

from keras.models import Sequential
from keras.layers import LSTM, Dense

model = Sequential()
model.add(LSTM(256, return_sequences = True, input_shape = (trainx.shape[1], 2)))
model.add(LSTM(128,input_shape = (trainx.shape[1], 2)))
model.add(Dense(2))
model.compile(loss = 'mean_squared_error', optimizer = 'adam')
model.fit(trainx, trainy, epochs = 2000, batch_size = 10, verbose = 2, shuffle = False)
model.save_weights('LSTMBasic1.h5')

In [407]:

model.load_weights('LSTMBasic1.h5')
predict = model.predict(testx)

Now let’s see what our predictions look like.

In [408]:

plt.plot(testx.reshape(398,2)[:,0:1], testx.reshape(398,2)[:,1:2])
plt.plot(predict[:,0:1], predict[:,1:2])

Out[408]:

[<matplotlib.lines.Line2D at 0x1eac792f048>]

Now, we should try and model a sine or cosine wave in a similar fashion. You can run the code given below and play with the model parameters to see how the results change.

In [409]:

x = numpy.arange (1,500,1)
y = numpy.sin(x)
plt.plot(x,y)

Out[409]:

[<matplotlib.lines.Line2D at 0x1eac7a0b3c8>]

In [410]:

trainx, testx = x[0:int(0.8*(len(x)))], x[int(0.8*(len(x))):]
trainy, testy = y[0:int(0.8*(len(y)))], y[int(0.8*(len(y))):]
train = numpy.array(list(zip(trainx,trainy)))
test = numpy.array(list(zip(trainx,trainy)))

In [411]:

look_back = 1
trainx,trainy = create_dataset(train, look_back)
testx,testy = create_dataset(test, look_back)
trainx = numpy.reshape(trainx, (trainx.shape[0], 1, 2))
testx = numpy.reshape(testx, (testx.shape[0], 1, 2))

In [ ]:

model = Sequential()
model.add(LSTM(512, return_sequences = True, input_shape = (trainx.shape[1], 2)))
model.add(LSTM(256,input_shape = (trainx.shape[1], 2)))
model.add(Dense(2))
model.compile(loss = 'mean_squared_error', optimizer = 'adam')
model.fit(trainx, trainy, epochs = 2000, batch_size = 10, verbose = 2, shuffle = False)
model.save_weights('LSTMBasic2.h5')

In [413]:

model.load_weights('LSTMBasic2.h5')
predict = model.predict(testx)

In [415]:

plt.plot(trainx.reshape(398,2)[:,0:1], trainx.reshape(398,2)[:,1:2])
plt.plot(predict[:,0:1], predict[:,1:2])

Out [415]:

[<matplotlib.lines.Line2D at 0x1eac7a1f550>]

Now you are ready to move on to any dataset.

It is important for us to quantify the performance of a model to use it as a feedback and comparison. In this tutorial we have used one of the most popular error metric root mean squared error. There are various other error metrics available. This chapter discusses them in brief.

Mean Square Error

It is the average of square of difference between the predicted values and true values. Sklearn provides it as a function. It has the same units as the true and predicted values squared and is always positive.

$$MSE = \frac{1}{n} \displaystyle\sum\limits_{t=1}^n \lgroup y'_{t}\:-y_{t}\rgroup^{2}$$

어디 $y'_{t}$ 예측 값입니다.

$y_{t}$ 실제 값이고

n은 테스트 세트의 총 값 수입니다.

방정식에서 MSE가 더 큰 오류 또는 특이 치에 대해 더 많은 페널티를주는 것이 분명합니다.

제곱 평균 제곱근 오차

평균 제곱 오차의 제곱근입니다. 또한 항상 양수이며 데이터 범위 내에 있습니다.

$$RMSE = \sqrt{\frac{1}{n} \displaystyle\sum\limits_{t=1}^n \lgroup y'_{t}-y_{t}\rgroup ^2}$$

어디, $y'_{t}$ 예측값

$y_{t}$ 실제 값이고

n은 테스트 세트의 총 값 수입니다.

그것은 통합의 힘에 있으므로 MSE에 비해 더 해석 가능합니다. RMSE는 또한 더 큰 오류에 대해 더 많은 불이익을줍니다. 튜토리얼에서 RMSE 메트릭을 사용했습니다.

평균 절대 오차

예측값과 참값의 절대 차이의 평균입니다. 예측 및 참 값과 동일한 단위를 가지며 항상 양수입니다.

$$MAE = \frac{1}{n}\displaystyle\sum\limits_{t=1}^{t=n} | y'{t}-y_{t}\lvert$$

어디, $y'_{t}$ 예측 가치,

$y_{t}$ 실제 값이고

n은 테스트 세트의 총 값 수입니다.

평균 백분율 오류

예측 값과 실제 값 사이의 절대 차이 평균을 실제 값으로 나눈 비율입니다.

$$MAPE = \frac{1}{n}\displaystyle\sum\limits_{t=1}^n\frac{y'_{t}-y_{t}}{y_{t}}*100\: \%$$

어디, $y'_{t}$ 예측 가치,

$y_{t}$ 실제 값이고 n은 테스트 세트의 총 값 수입니다.

그러나이 오류를 사용할 때의 단점은 양의 오류와 음의 오류가 서로 상쇄 될 수 있다는 것입니다. 따라서 절대 백분율 오류가 사용됩니다.

평균 절대 백분율 오류

예측 값과 실제 값 사이의 절대 차이 평균을 실제 값으로 나눈 비율입니다.

$$MAPE = \frac{1}{n}\displaystyle\sum\limits_{t=1}^n\frac{|y'_{t}-y_{t}\lvert}{y_{t}}*100\: \%$$

어디 $y'_{t}$ 예측값

$y_{t}$ 실제 값이고

n은 테스트 세트의 총 값 수입니다.

이 자습서에서 시계열 분석에 대해 논의했습니다.이를 통해 시계열 모델이 먼저 기존 관측치에서 추세와 계절성을 인식 한 다음이 추세와 계절성을 기반으로 값을 예측한다는 이해를 얻었습니다. 이러한 분석은 다음과 같은 다양한 분야에서 유용합니다.

  • Financial Analysis − 여기에는 판매 예측, 재고 분석, 주식 시장 분석, 가격 추정이 포함됩니다.

  • Weather Analysis − 여기에는 온도 추정, 기후 변화, 계절 변화 인식, 일기 예보가 포함됩니다.

  • Network Data Analysis − 여기에는 네트워크 사용량 예측, 이상 또는 침입 감지, 예측 유지 보수가 포함됩니다.

  • Healthcare Analysis − 인구 조사 예측, 보험 혜택 예측, 환자 모니터링이 포함됩니다.

기계 학습은 다양한 종류의 문제를 다룹니다. 실제로 거의 모든 분야에는 기계 학습의 도움으로 자동화되거나 개선 될 수있는 범위가 있습니다. 많은 작업이 수행되는 몇 가지 문제가 아래에 나와 있습니다.

시계열 데이터

이것은 시간에 따라 변하는 데이터이므로 시간이 중요한 역할을합니다.이 튜토리얼에서 주로 논의했습니다.

비시 계열 데이터

시간과 무관 한 데이터이며 ML 문제의 주요 비율은 비시 계열 데이터에 있습니다. 간단하게하기 위해 다음과 같이 추가로 분류합니다.

  • Numerical Data − 컴퓨터는 인간과 달리 숫자 만 이해하므로 모든 종류의 데이터는 궁극적으로 머신 러닝을 위해 숫자 데이터로 변환됩니다. 예를 들어 이미지 데이터는 (r, b, g) 값으로 변환되고 문자는 ASCII 코드 또는 단어로 변환됩니다. 숫자로 인덱싱되고 음성 데이터는 숫자 데이터를 포함하는 mfcc 파일로 변환됩니다.

  • Image Data − 컴퓨터 비전은 컴퓨터 세계에 혁명을 일으켰으며 의학, 위성 영상 등 다양한 분야에서 응용되고 있습니다.

  • Text Data− NLP (Natural Language Processing)는 텍스트 분류, 패러 프레이즈 감지 및 언어 요약에 사용됩니다. 이것이 Google과 Facebook을 스마트하게 만드는 것입니다.

  • Speech Data− 음성 처리에는 음성 인식 및 감정 이해가 포함됩니다. 컴퓨터에 인간과 같은 특성을 부여하는 데 중요한 역할을합니다.