Les fonctions de série temporelle sont des fonctions d'agrégation qui opèrent sur des séquences de valeurs de données mesurées en points dans le temps.
Les sections ci-après décrivent certaines fonctions de série temporelle disponibles dans différents packages de série temporelle.
Transformations
Les transformations sont des fonctions appliquées sur une série temporelle pour générer une autre série temporelle. La bibliothèque de séries temporelles prend en charge différents types de transformation, y compris les transformations fournies (à l'aide de from tspy.functions import transformers
), ainsi que les transformations définies par l'utilisateur.
L'exemple suivant présente certaines transformations fournies :
#Interpolation
>>> ts = tspy.time_series([1.0, 2.0, 3.0, 4.0, 5.0, 6.0])
>>> periodicity = 2
>>> interp = interpolators.nearest(0.0)
>>> interp_ts = ts.resample(periodicity, interp)
>>> interp_ts.print()
TimeStamp: 0 Value: 1.0
TimeStamp: 2 Value: 3.0
TimeStamp: 4 Value: 5.0
#Fillna
>>> shift_ts = ts.shift(2)
print("shifted ts to add nulls")
print(shift_ts)
print("\nfilled ts to make nulls 0s")
null_filled_ts = shift_ts.fillna(interpolators.fill(0.0))
print(null_filled_ts)
shifted ts to add nulls
TimeStamp: 0 Value: null
TimeStamp: 1 Value: null
TimeStamp: 2 Value: 1.0
TimeStamp: 3 Value: 2.0
TimeStamp: 4 Value: 3.0
TimeStamp: 5 Value: 4.0
filled ts to make nulls 0s
TimeStamp: 0 Value: 0.0
TimeStamp: 1 Value: 0.0
TimeStamp: 2 Value: 1.0
TimeStamp: 3 Value: 2.0
TimeStamp: 4 Value: 3.0
TimeStamp: 5 Value: 4.0
# Additive White Gaussian Noise (AWGN)
>>> noise_ts = ts.transform(transformers.awgn(mean=0.0,sd=.03))
>>> print(noise_ts)
TimeStamp: 0 Value: 0.9962378841388397
TimeStamp: 1 Value: 1.9681980879378596
TimeStamp: 2 Value: 3.0289374962174405
TimeStamp: 3 Value: 3.990728648807705
TimeStamp: 4 Value: 4.935338359740761
TimeStamp: 5 Value: 6.03395072999318
Segmentation
La segmentation ou le fenêtrage est le processus de fractionnement d'une série temporelle en plusieurs segments. La bibliothèque de séries temporelles prend en charge diverses formes de segmentation et permet également de créer des segments définis par l'utilisateur.
Segmentation basée sur une fenêtre
Ce type de segmentation d'une série temporelle est basé sur des tailles de segment spécifiées par l'utilisateur. Les segments peuvent être basés sur des enregistrements ou sur le temps. Des options permettent de créer des segments basés sur des fenêtres en cascade ou coulissantes.
>>> import tspy >>> ts_orig = tspy.builder() .add(tspy.observation(1,1.0)) .add(tspy.observation(2,2.0)) .add(tspy.observation(6,6.0)) .result().to_time_series() >>> ts_orig timestamp: 1 Value: 1.0 timestamp: 2 Value: 2.0 timestamp: 6 Value: 6.0 >>> ts = ts_orig.segment_by_time(3,1) >>> ts timestamp: 1 Value: original bounds: (1,3) actual bounds: (1,2) observations: [(1,1.0),(2,2.0)] timestamp: 2 Value: original bounds: (2,4) actual bounds: (2,2) observations: [(2,2.0)] timestamp: 3 Value: this segment is empty timestamp: 4 Value: original bounds: (4,6) actual bounds: (6,6) observations: [(6,6.0)]
Segmentation basé sur l'ancrage
La segmentation par ancrage est un type de segmentation très important qui crée un segment en ancrant un lambda spécifique, qui peut être une valeur simple. Par exemple, vous pouvez observer les événements ayant précédé une erreur 500 ou examiner les valeurs après avoir observé une anomalie. Les variantes de la segmentation par ancrage incluent la spécification d'une plage avec plusieurs repères.
>>> import tspy >>> ts_orig = tspy.time_series([1.0, 2.0, 3.0, 4.0, 5.0]) >>> ts_orig timestamp: 0 Value: 1.0 timestamp: 1 Value: 2.0 timestamp: 2 Value: 3.0 timestamp: 3 Value: 4.0 timestamp: 4 Value: 5.0 >>> ts = ts_orig.segment_by_anchor(lambda x: x % 2 == 0, 1, 2) >>> ts timestamp: 1 Value: original bounds: (0,3) actual bounds: (0,3) observations: [(0,1.0),(1,2.0),(2,3.0),(3,4.0)] timestamp: 3 Value: original bounds: (2,5) actual bounds: (2,4) observations: [(2,3.0),(3,4.0),(4,5.0)]
Outils de segmentation
Il existe plusieurs outils de segmentation spécialisés prêts à l'emploi en important le package
segmenters
(à l'aide defrom tspy.functions import segmenters
). Un exemple d'outil de segmentation est un outil qui utilise la régression pour segmenter une série temporelle :>>> ts = tspy.time_series([1.0,2.0,3.0,4.0,5.0,2.0,1.0,-1.0,50.0,53.0,56.0]) >>> max_error = .5 >>> skip = 1 >>> reg_sts = ts.to_segments(segmenters.regression(max_error,skip,use_relative=True)) >>> reg_sts timestamp: 0 Value: range: (0, 4) outliers: {} timestamp: 5 Value: range: (5, 7) outliers: {} timestamp: 8 Value: range: (8, 10) outliers: {}
Réducteurs
Un réducteur est une fonction appliquée aux valeurs d'un ensemble de séries temporelles pour générer une valeur unique. Les fonctions reducer
de la série temporelle sont similaires au concept de réducteur utilisé par Hadoop/Spark. Cette valeur unique peut être une collection, mais il s'agit plus généralement d'un objet unique. Un exemple de fonction de réducteur consiste à calculer la moyenne des valeurs d'une série temporelle.
Plusieurs fonctions reducer
sont prises en charge, notamment :
Réducteurs de distance
Les réducteurs de distance sont une classe de réducteurs qui calculent la distance entre deux séries temporelles. La bibliothèque prend en charge les fonctions de distance numériques et catégorielles sur les séquences. Cela inclut les mesures de distance de déformation temporelle, telles que les contraintes Itakura Parallelogram, Sakoe-Chiba Band, DTW sans contrainte et DTW non déformées dans le temps. Les distances de distribution telles que la distance hongroise et la distance du terrassier sont également disponibles.
Pour les mesures de distance des séries temporelles catégorielles, vous pouvez utiliser les mesures de distance Damerau Levenshtein et Jaro-Winkler.
>>> from tspy.functions import * >>> ts = tspy.time_series([1.0, 2.0, 3.0, 4.0, 5.0, 6.0]) >>> ts2 = ts.transform(transformers.awgn(sd=.3)) >>> dtw_distance = ts.reduce(ts2,reducers.dtw(lambda obs1, obs2: abs(obs1.value - obs2.value))) >>> print(dtw_distance) 1.8557981638880405
Réducteurs mathématiques
Plusieurs réducteurs mathématiques pratiques sont fournis pour les séries temporelles numériques. Ils incluent des réducteurs de base, tels que la moyenne, la somme, l'écart type et les moments. L'entropie, le kurtosis, la FFT et ses variantes, diverses corrélations et un histogramme sont également inclus. Un réducteur de synthèse de base pratique est la fonction
describe
qui fournit des informations de base sur la série temporelle.>>> from tspy.functions import * >>> ts = tspy.time_series([1.0, 2.0, 3.0, 4.0, 5.0, 6.0]) >>> ts2 = ts.transform(transformers.awgn(sd=.3)) >>> corr = ts.reduce(ts2, reducers.correlation()) >>> print(corr) 0.9938941942380525 >>> adf = ts.reduce(reducers.adf()) >>> print(adf) pValue: -3.45 satisfies test: false >>> ts2 = ts.transform(transformers.awgn(sd=.3)) >>> granger = ts.reduce(ts2, reducers.granger(1)) >>> print(granger) #f_stat, p_value, R2 -1.7123613937876463,-3.874412217575385,1.0
Un autre réducteur de base très utile pour bien comprendre les séries temporelles est le réducteur de description. Ce réducteur est illustré ci-dessous :
>>> desc = ts.describe() >>> print(desc) min inter-arrival-time: 1 max inter-arrival-time: 1 mean inter-arrival-time: 1.0 top: null unique: 6 frequency: 1 first: TimeStamp: 0 Value: 1.0 last: TimeStamp: 5 Value: 6.0 count: 6 mean:3.5 std:1.707825127659933 min:1.0 max:6.0 25%:1.75 50%:3.5 75%:5.25
Jointures temporelles
La bibliothèque inclut des fonctions pour les jointures temporelles ou la jointure de séries temporelles en fonction de leur horodatage. Les fonctions de jointure sont similaires à celles d'une base de données (jointures à gauche, à droite, externes, internes, externes sur l'élément de gauche, externes sur l'élément de droite, etc.) Les exemples de code suivants présentent certaines de ces fonctions de jointure :
# Create a collection of observations (materialized TimeSeries)
observations_left = tspy.observations(tspy.observation(1, 0.0), tspy.observation(3, 1.0), tspy.observation(8, 3.0), tspy.observation(9, 2.5))
observations_right = tspy.observations(tspy.observation(2, 2.0), tspy.observation(3, 1.5), tspy.observation(7, 4.0), tspy.observation(9, 5.5), tspy.observation(10, 4.5))
# Build TimeSeries from Observations
ts_left = observations_left.to_time_series()
ts_right = observations_right.to_time_series()
# Perform full join
ts_full = ts_left.full_join(ts_right)
print(ts_full)
TimeStamp: 1 Value: [0.0, null]
TimeStamp: 2 Value: [null, 2.0]
TimeStamp: 3 Value: [1.0, 1.5]
TimeStamp: 7 Value: [null, 4.0]
TimeStamp: 8 Value: [3.0, null]
TimeStamp: 9 Value: [2.5, 5.5]
TimeStamp: 10 Value: [null, 4.5]
# Perform left align with interpolation
ts_left_aligned, ts_right_aligned = ts_left.left_align(ts_right, interpolators.nearest(0.0))
print("left ts result")
print(ts_left_aligned)
print("right ts result")
print(ts_right_aligned)
left ts result
TimeStamp: 1 Value: 0.0
TimeStamp: 3 Value: 1.0
TimeStamp: 8 Value: 3.0
TimeStamp: 9 Value: 2.5
right ts result
TimeStamp: 1 Value: 0.0
TimeStamp: 3 Value: 1.5
TimeStamp: 8 Value: 4.0
TimeStamp: 9 Value: 5.5
Prévision
Une fonctionnalité clé fournie par la bibliothèque de séries temporelles est la prévision. La bibliothèque comprend des fonctions pour des modèles de prévision simples et complexes, notamment ARIMA, Exponential, Holt-Winters et BATS. L'exemple suivant présente la fonction permettant de créer une prévision Holt-Winters :
import random
model = tspy.forecasters.hws(samples_per_season=samples_per_season, initial_training_seasons=initial_training_seasons)
for i in range(100):
timestamp = i
value = random.randint(1,10) * 1.0
model.update_model(timestamp, value)
print(model)
Forecasting Model
Algorithm: HWSAdditive=5 (aLevel=0.001, bSlope=0.001, gSeas=0.001) level=6.087789839896166, slope=0.018901997884893912, seasonal(amp,per,avg)=(1.411203455586738,5, 0,-0.0037471500727535465)
#Is model init-ed
if model.is_initialized():
print(model.forecast_at(120))
6.334135728495107
ts = tspy.time_series([float(i) for i in range(10)])
print(ts)
TimeStamp: 0 Value: 0.0
TimeStamp: 1 Value: 1.0
TimeStamp: 2 Value: 2.0
TimeStamp: 3 Value: 3.0
TimeStamp: 4 Value: 4.0
TimeStamp: 5 Value: 5.0
TimeStamp: 6 Value: 6.0
TimeStamp: 7 Value: 7.0
TimeStamp: 8 Value: 8.0
TimeStamp: 9 Value: 9.0
num_predictions = 5
model = tspy.forecasters.auto(8)
confidence = .99
predictions = ts.forecast(num_predictions, model, confidence=confidence)
print(predictions.to_time_series())
TimeStamp: 10 Value: {value=10.0, lower_bound=10.0, upper_bound=10.0, error=0.0}
TimeStamp: 11 Value: {value=10.997862810553725, lower_bound=9.934621260488143, upper_bound=12.061104360619307, error=0.41277640121597475}
TimeStamp: 12 Value: {value=11.996821082897318, lower_bound=10.704895525154571, upper_bound=13.288746640640065, error=0.5015571318964149}
TimeStamp: 13 Value: {value=12.995779355240911, lower_bound=11.50957896664928, upper_bound=14.481979743832543, error=0.5769793776877866}
TimeStamp: 14 Value: {value=13.994737627584504, lower_bound=12.33653268707341, upper_bound=15.652942568095598, error=0.6437557559526337}
print(predictions.to_time_series().to_df())
timestamp value lower_bound upper_bound error
0 10 10.000000 10.000000 10.000000 0.000000
1 11 10.997863 9.934621 12.061104 0.412776
2 12 11.996821 10.704896 13.288747 0.501557
3 13 12.995779 11.509579 14.481980 0.576979
4 14 13.994738 12.336533 15.652943 0.643756
SQL de série temporelle
La bibliothèque de séries temporelles est étroitement intégrée à Apache Spark. A l'aide des nouveaux types de données de Spark Catalyst, vous pouvez effectuer des opérations SQL de série temporelle qui se développent horizontalement à l'aide d'Apache Spark. Cela vous permet d'utiliser facilement des extensions de séries temporelles dans IBM Analytics Engine ou dans des solutions qui incluent des fonctionnalités d'IBM Analytics Engine comme les environnements watsonx.ai Studio Spark.
Les extensions SQL couvrent la plupart des aspects des fonctions de série temporelle, y compris la segmentation, les transformations, les réducteurs, les prévisions et les E-S. Voir Analyse des données de série temporelle.
En savoir plus
Pour utiliser le kit SDK tspy
Python, voir la documentation du kit SDK tspy
Python.