path = window.location.pathname.replace(".html", ".qmd");
path_modified = (path.includes('en/content')) ? path.replace('en/content', 'content/en') : path
html`${printBadges({fpath: path_modified})}`
Ce chapitre présente la première application d’une journée de cours que j’ai donné à l’Université Dauphine dans le cadre des PSL Data Week.
Dérouler les _slides_ associées ci-dessous ou [cliquer ici](https://linogaliana.github.io/dauphine-week-data/#/title-slide) pour les afficher en plein écran.
Pour lire les données de manière efficace, nous
proposons d’utiliser le package duckdb
.
Pour l’installer, voici la commande :
!pip install duckdb
1 Pourquoi utiliser les pipelines ?
1.1 Définitions préalables
Ce chapitre nous amènera à explorer plusieurs écosystèmes, pour lesquels on retrouve quelques buzz-words dont voici les définitions :
Terme | Définition |
---|---|
DevOps | Mouvement en ingénierie informatique et une pratique technique visant à l’unification du développement logiciel (dev) et de l’administration des infrastructures informatiques (ops) |
MLOps | Ensemble de pratiques qui vise à déployer et maintenir des modèles de machine learning en production de manière fiable et efficace |
Ce chapitre fera des références régulières au cours de 3e année de l’ENSAE “Mise en production de projets data science”.
1.2 Objectif
Les chapitres précédents ont permis de montrer des bouts de code épars pour entraîner des modèles ou faire du preprocessing. Cette démarche est intéressante pour tâtonner mais risque d’être coûteuse ultérieurement s’il est nécessaire d’ajouter une étape de preprocessing ou de changer d’algorithme.
Les pipelines sont pensés pour simplifier la mise en production ultérieure d’un modèle de machine learning. Ils sont au coeur de la démarche de MLOps qui est présentée dans le cours de 3e année de l’ENSAE de “Mise en production de projets data science”, qui vise à simplifier la mise en oeuvre opérationnelle de projets utilisant des techniques de machine learning.
import matplotlib.pyplot as plt
import seaborn as sns
import numpy as np
import pandas as pd
1.3 Les pipelines Scikit
Heureusement, Scikit
propose un excellent outil pour proposer un cadre
général pour créer une chaîne de production machine learning. Il
s’agit des
pipelines.
Ils présentent de nombreux intérêts, parmi lesquels :
- Ils sont très pratiques et lisibles. On rentre des données en entrée, on n’appelle qu’une seule fois les méthodes
fit
etpredict
ce qui permet de s’assurer une gestion cohérente des transformations de variables, par exemple après l’appel d’unStandardScaler
; - La modularité rend aisée la mise à jour d’un pipeline et renforce la capacité à le réutiliser ;
- Ils permettent de facilement chercher les hyperparamètres d’un modèle. Sans pipeline, écrire un code qui fait du tuning d’hyperparamètres peut être pénible. Avec les pipelines, c’est une ligne de code ;
- La sécurité d’être certain que les étapes de preprocessing sont bien appliquées aux jeux de données désirés avant l’estimation.
Hint
Un des intérêts des pipelines scikit est qu’ils fonctionnent aussi avec
des méthodes qui ne sont pas issues de scikit
.
Il est possible d’introduire un modèle de réseau de neurone Keras
dans
un pipeline scikit
.
Pour introduire un modèle économétrique statsmodels
c’est un peu plus coûteux mais nous allons proposer des exemples
qui peuvent servir de modèle et qui montrent que c’est faisable
sans trop de difficulté.
3 Un premier pipeline : random forest sur des variables standardisées
Notre premier pipeline va nous permettre d’intégrer ensemble:
- Une étape de preprocessing avec la standardisation de variables
- Une étape d’estimation du prix en utilisant un modèle de random forest
Pour le moment, on va prendre comme acquis un certain nombre de variables explicatives (les features) et les hyperparamètres du modèle.
L’algorithme des random forest est une technique statistique basée sur les arbres de décision. Elle a été définie explicitement par l’un des pionniers du machine learning, Breiman (2001). Il s’agit d’une méthode ensembliste puisqu’elle consiste à utiliser plusieurs algorithmes (en l’occurrence des arbres de décision) pour obtenir une meilleure prédiction que ne le permettraient chaque modèle isolément.
Les random forest sont une méthode d’aggrégation1 d’arbres de décision. On calcule \(K\) arbres de décision et en tire, par une méthode d’agrégation, une règle de décision moyenne qu’on va appliquer pour tirer une prédiction de nos données.
L’un des intérêts des random forest est qu’il existe des méthodes pour déterminer l’importance relative de chaque variable dans la prédiction.
Nous allons ici partir d’un random forest avec des valeurs d’hyperparamètres données, à savoir la profondeur de l’arbre.
3.1 Définition des ensembles train et test
Nous allons donc nous restreindre à un sous-ensemble de colonnes dans un premier temps.
Nous allons également ne conserver que les transactions inférieures à 5 millions d’euros (on anticipe que celles ayant un montant supérieur sont des transactions exceptionnelles dont le mécanisme de fixation du prix diffère)
= mutations.drop(
mutations2
colonnes_surface.tolist()+ ["Date mutation", "lprix"], # ajouter "confinement" si données 2020
="columns",
axis
).copy()
= mutations2.loc[
mutations2 "Valeur fonciere"] < 5e6
mutations2[# keep only values below 5 millions
]
= mutations2.columns.str.replace(" ", "_")
mutations2.columns = mutations2.dropna(subset=["dep", "Code_type_local", "month"]) mutations2
Notre pipeline va incorporer deux types de variables: les variables catégorielles et les variables numériques. Ces différents types vont bénéficier d’étapes de preprocessing différentes.
= mutations2.columns[
numeric_features ~mutations2.columns.isin(["dep", "Code_type_local", "month", "Valeur_fonciere"])
].tolist()= ["dep", "Code_type_local", "month"] categorical_features
Au passage, nous avons abandonné la variable de code postal pour privilégier le département afin de réduire la dimension de notre jeu de données. Si on voulait vraiment avoir un bon modèle, il faudrait faire autrement car le code postal est probablement un très bon prédicteur du prix d’un bien, une fois que les caractéristiques du bien sont contrôlées.
Exercice 1 : Découpage des échantillons
Nous allons stratifier notre échantillonage de train/test par département afin de tenir compte, de manière minimale, de la géographie. Pour accélérer les calculs pour ce tutoriel, nous n’allons considérer que 30% des transactions observées sur chaque département.
Voici le code pour le faire:
= mutations2.groupby("dep").sample(frac=0.1, random_state=123) mutations2
Avec la fonction adéquate de Scikit
, faire un découpage de mutations2
en train et test sets
en suivant les consignes suivantes:
- 20% des données dans l’échantillon de test ;
- L’échantillonnage est stratifié par départements ;
- Pour avoir des résultats reproductibles, choisir une racine égale à 123.
3.2 Définition du premier pipeline
Pour commencer, nous allons fixer la taille des arbres de décision avec
l’hyperparamètre max_depth = 2
.
Notre pipeline va intégrer les étapes suivantes :
- Preprocessing :
- Les variables numériques vont être standardisées avec un
StandardScaler
. Pour cela, nous allons utiliser la listenumeric_features
définie précédemment. - Les variables catégorielles vont être explosées avec un one hot encoding
(méthode
OneHotEncoder
descikit
) Pour cela, nous allons utiliser la listecategorical_features
- Les variables numériques vont être standardisées avec un
- Random forest : nous allons appliquer l’estimateur ad hoc de
Scikit
.
Exercice 2 : Construction d'un premier pipeline formel
- Initialiser un random forest de profondeur 2. Fixer la racine à 123 pour avoir des résultats reproductibles.
- La première étape du pipeline (nommer cette couche preprocessor) consiste à appliquer les étapes de preprocessing adaptées à chaque type de variables:
- Pour les variables numériques, appliquer une étape d’imputation à la moyenne puis standardiser celles-ci
- Pour les variables catégorielles, appliquer un one hot encoding
- Appliquer comme couche de sortie le modèle défini plus tôt.
💡 Il est recommandé de s’aider de la documentation de Scikit
. Si vous avez besoin d’un indice supplémentaire, consulter le pipeline présenté ci-dessous.
A l’issue de cet exercice, nous devrions obtenir le pipeline suivant.
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['dep', 'Code_type_local', 'month'])])), ('randomforest', RandomForestRegressor(max_depth=2, random_state=123))])In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['dep', 'Code_type_local', 'month'])])), ('randomforest', RandomForestRegressor(max_depth=2, random_state=123))])
ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['dep', 'Code_type_local', 'month'])])
['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']
SimpleImputer()
StandardScaler()
['dep', 'Code_type_local', 'month']
OneHotEncoder(handle_unknown='ignore', sparse_output=False)
RandomForestRegressor(max_depth=2, random_state=123)
Nous avons construit ce pipeline sous forme de couches successives. La couche
randomforest
prendra automatiquement le résultat de la couche preprocessor
en input. La couche features
permet d’introduire de manière relativement
simple (quand on a les bonnes méthodes) la complexité du preprocessing
sur données réelles dont les types divergent.
A cette étape, rien n’a encore été estimé. C’est très simple à mettre en oeuvre avec un pipeline.
Exercice 3 : Mise en oeuvre du pipeline
- Estimer les paramètres du modèle sur le jeu d’entraînement
- Observer la manière dont les données d’entraînement sont transformées
par l’étape de preprocessing avec les méthodes adéquates sur 4 observations de
X_train
tirées aléatoirement - Utiliser ce modèle pour prédire le prix sur l’échantillon de test. A partir de ces quelques prédictions, quel semble être le problème ?
- Observer la manière dont ce preprocessing peut s’appliquer sur deux exemples fictifs :
- Un appartement (
code_type_local = 2
) dans le 75, vendu au mois de mai, unique lot de la vente avec 3 pièces, faisant 75m² ; - Une maison (
code_type_local = 1
) dans le 06, vendue en décembre, dans une transaction avec 2 lots. La surface complète est de 180m² et le bien comporte 6 pièces.
- Un appartement (
- Déduire sur ces deux exemples le prix prédit par le modèle.
- Calculer et interpréter le RMSE sur l’échantillon de test. Ce modèle est-il satisfaisant ?
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['dep', 'Code_type_local', 'month'])])), ('randomforest', RandomForestRegressor(max_depth=2, random_state=123))])In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['dep', 'Code_type_local', 'month'])])), ('randomforest', RandomForestRegressor(max_depth=2, random_state=123))])
ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['dep', 'Code_type_local', 'month'])])
['Nombre_de_lots', 'Nombre_pieces_principales', 'surface']
SimpleImputer()
StandardScaler()
['dep', 'Code_type_local', 'month']
OneHotEncoder(handle_unknown='ignore', sparse_output=False)
RandomForestRegressor(max_depth=2, random_state=123)
array([282871.63598981, 301165.65351098, 301165.65351098, ...,
282871.63598981, 471048.40037679, 282871.63598981])
array([642280.20111587, 282871.63598981])
433497.6437239088
3.3 Variable importance
Les prédictions semblent avoir une assez faible variance, comme si des variables de seuils intervenaient. Nous allons donc devoir essayer de comprendre pourquoi.
La “variable importance” se réfère à la mesure de l’influence de chaque variable d’entrée sur la performance du modèle. L’impureté fait référence à l’incertitude ou à l’entropie présente dans un ensemble de données. Dans le contexte des random forest, cette mesure est souvent calculée en évaluant la réduction moyenne de l’impureté des nœuds de décision causée par une variable spécifique. Cette approche permet de quantifier l’importance des variables dans le processus de prise de décision du modèle, offrant ainsi des intuitions sur les caractéristiques les plus informatives pour la prédiction (plus de détails sur ce blog).
On ne va représenter, parmi notre ensemble important de colonnes, que celles qui ont une importance non nulle.
Exercice 4 : Compréhension du modèle
- Récupérer la feature importance directement depuis la couche adaptée de votre pipeline
- Utiliser le code suivant pour calculer l’intervalle de confiance de cette mesure d’importance:
= np.std(
std for tree in pipe["randomforest"].estimators_], axis=0
[tree.feature_importances_ )
- Représenter les variables d’importance non nulle. Qu’en concluez-vous ?
Le graphique d’importance des variables que vous devriez obtenir à l’issue de cet exercice est le suivant.
<Axes: title={'center': 'Feature importances using MDI'}, ylabel='Mean decrease in impurity'>
Les statistiques obtenues par le biais de cette variable importance sont un peu rudimentaires mais permettent déjà de comprendre le problème de notre modèle.
On voit donc que deux de nos variables déterminantes sont des effets fixes géographiques (qui servent à ajuster de la différence de prix entre Paris et les Hauts de Seine et le reste de la France), une autre variable est un effet fixe type de bien. Les deux variables qui pourraient introduire de la variabilité, à savoir la surface et, dans une moindre mesure, le nombre de lots, ont une importance moindre.
Note
Idéalement, on utiliserait Yellowbrick
pour représenter l’importance des variables
Mais en l’état actuel du pipeline on a beaucoup de variables dont le poids
est nul qui viennent polluer la visualisation. Vous pouvez
consulter la
documentation de Yellowbrick
sur ce sujet
Les prédictions peuvent nous suggérer également qu’il y a un problème.
4 Restriction du champ du modèle
Mettre en oeuvre un bon modèle de prix au niveau France entière est complexe. Nous allons donc nous restreindre au champ suivant: les appartements dans Paris.
= mutations.drop(
mutations_paris
colonnes_surface.tolist()+ ["Date mutation", "lprix"], # ajouter "confinement" si données 2020
="columns",
axis
).copy()
= mutations_paris.loc[
mutations_paris "Valeur fonciere"] < 5e6
mutations_paris[# keep only values below 5 millions
]
= mutations_paris.columns.str.replace(" ", "_")
mutations_paris.columns = mutations_paris.dropna(subset=["dep", "Code_type_local", "month"])
mutations_paris = mutations_paris.loc[mutations_paris["dep"] == "75"]
mutations_paris = mutations_paris.loc[mutations_paris["Code_type_local"] == 2].drop(
mutations_paris "dep", "Code_type_local"], axis="columns"
[
)"surface"] > 0] mutations_paris.loc[mutations_paris[
month | Valeur_fonciere | Nombre_de_lots | Nombre_pieces_principales | surface | |
---|---|---|---|---|---|
4272176 | 1 | 686000.0 | 1 | 4.0 | 73.45 |
4273807 | 2 | 400000.0 | 1 | 2.0 | 41.16 |
4275450 | 3 | 475000.0 | 3 | 2.0 | 47.7 |
4275883 | 2 | 396200.0 | 2 | 2.0 | 47.14 |
4275884 | 2 | 299500.0 | 1 | 1.0 | 34.02 |
... | ... | ... | ... | ... | ... |
4617521 | 12 | 1650000.0 | 3 | 6.0 | 154.75 |
4617532 | 12 | 525000.0 | 2 | 3.0 | 46.65 |
4617540 | 12 | 425000.0 | 2 | 1.0 | 27.85 |
4617549 | 12 | 330000.0 | 2 | 2.0 | 34.32 |
4617577 | 3 | 232650.0 | 1 | 1.0 | 28.25 |
27971 rows × 5 columns
Exercice 4 : Pipeline plus simple
Reprendre les codes précédents et reconstruire notre pipeline sur la nouvelle base en mettant en oeuvre une méthode de boosting plutôt qu’une forêt aléatoire.
La correction de cet exercice est apparente pour simplifier les prochaines étapes mais essayez de faire celui-ci de vous-même.
A l’issue de cet exercice, vous devriez avoir des MDI proches de celles-ci :
from sklearn.ensemble import GradientBoostingRegressor
from sklearn.impute import SimpleImputer
from sklearn.preprocessing import StandardScaler, OneHotEncoder
from sklearn.pipeline import make_pipeline, Pipeline
from sklearn.compose import make_column_transformer
from sklearn.model_selection import train_test_split
= mutations.drop(
mutations_paris
colonnes_surface.tolist()+ ["Date mutation", "lprix"], # ajouter "confinement" si données 2020
="columns",
axis
).copy()
= mutations_paris.loc[
mutations_paris "Valeur fonciere"] < 5e6
mutations_paris[# keep only values below 5 millions
]
= mutations_paris.columns.str.replace(" ", "_")
mutations_paris.columns = mutations_paris.dropna(subset=["dep", "Code_type_local", "month"])
mutations_paris = mutations_paris.loc[mutations_paris["dep"] == "75"]
mutations_paris = mutations_paris.loc[mutations_paris["Code_type_local"] == 2].drop(
mutations_paris "dep", "Code_type_local", "Nombre_de_lots"], axis="columns"
[
)"surface"] > 0]
mutations_paris.loc[mutations_paris[
= mutations_paris.columns[
numeric_features ~mutations_paris.columns.isin(["month", "Valeur_fonciere"])
].tolist()= ["month"]
categorical_features
= GradientBoostingRegressor(random_state=0)
reg
= make_pipeline(SimpleImputer(), StandardScaler())
numeric_pipeline = make_column_transformer(
transformer
(numeric_pipeline, numeric_features),=False, handle_unknown="ignore"), categorical_features),
(OneHotEncoder(sparse_output
)= Pipeline(steps=[("preprocessor", transformer), ("boosting", reg)])
pipe
= train_test_split(
X_train, X_test, y_train, y_test "Valeur_fonciere", axis=1),
mutations_paris.drop("Valeur_fonciere"]].values.ravel(),
mutations_paris[[=0.2,
test_size=123,
random_state
)
pipe.fit(X_train, y_train)
pd.DataFrame("boosting"].feature_importances_, index=pipe[:-1].get_feature_names_out()
pipe[ )
0 | |
---|---|
pipeline__Nombre_pieces_principales | 0.268176 |
pipeline__surface | 0.716371 |
onehotencoder__month_1 | 0.000309 |
onehotencoder__month_2 | 0.001060 |
onehotencoder__month_3 | 0.000967 |
onehotencoder__month_4 | 0.007498 |
onehotencoder__month_5 | 0.000000 |
onehotencoder__month_6 | 0.000153 |
onehotencoder__month_7 | 0.000156 |
onehotencoder__month_8 | 0.000819 |
onehotencoder__month_9 | 0.000026 |
onehotencoder__month_10 | 0.002170 |
onehotencoder__month_11 | 0.001690 |
onehotencoder__month_12 | 0.000605 |
5 Recherche des hyperparamètres optimaux avec une validation croisée
On détecte que le premier modèle n’est pas très bon et ne nous aidera pas vraiment à évaluer de manière fiable l’appartement de nos rêves.
On va essayer de voir si notre modèle ne serait pas meilleur avec des hyperparamètres plus adaptés. Après tout, nous avons choisi par défaut la profondeur de l’arbre mais c’était un choix au doigt mouillé.
❓️ Quels sont les hyperparamètres qu’on peut essayer d’optimiser ?
"boosting"].get_params() pipe[
{'alpha': 0.9,
'ccp_alpha': 0.0,
'criterion': 'friedman_mse',
'init': None,
'learning_rate': 0.1,
'loss': 'squared_error',
'max_depth': 3,
'max_features': None,
'max_leaf_nodes': None,
'min_impurity_decrease': 0.0,
'min_samples_leaf': 1,
'min_samples_split': 2,
'min_weight_fraction_leaf': 0.0,
'n_estimators': 100,
'n_iter_no_change': None,
'random_state': 0,
'subsample': 1.0,
'tol': 0.0001,
'validation_fraction': 0.1,
'verbose': 0,
'warm_start': False}
Un détour par la documentation
nous aide à comprendre ceux sur lesquels on va jouer. Par exemple, il serait
absurde de jouer sur le paramètre random_state
qui est la racine du générateur
pseudo-aléatoire.
= pd.concat((X_train, X_test), axis=0)
X = np.concatenate([y_train, y_test]) Y
Nous allons nous contenter de jouer sur les paramètres:
n_estimators
: Le nombre d’arbres de décision que notre forêt contientmax_depth
: La profondeur de chaque arbre
Il existe plusieurs manières de faire de la validation croisée. Nous allons ici
utiliser la grid search qui consiste à estimer et tester le modèle sur chaque
combinaison d’une grille de paramètres et sélectionner le couple de valeurs
des hyperparamètres amenant à la meilleure prédiction. Par défaut, scikit
effectue une 5-fold cross validation. Nous n’allons pas changer
ce comportement.
Comme expliqué précédemment, les paramètres s’appelent sous la forme
<step>__<parameter_name>
La validation croisée pouvant être très consommatrice de temps, nous
n’allons l’effectuer que sur un nombre réduit de valeurs de notre grille.
Il est possible de passer la liste des valeurs à passer au crible sous
forme de liste
(comme nous allons le proposer pour l’argument max_depth
dans l’exercice ci-dessous) ou
sous forme d’array
(comme nous allons le proposer pour l’argument n_estimators
) ce qui est
souvent pratique pour générer un criblage d’un intervalle avec np.linspace
.
Hint
Les estimations sont, par défaut, menées de manière séquentielle (l’une après
l’autre). Nous sommes cependant face à un problème
embarassingly parallel.
Pour gagner en performance, il est recommandé d’utiliser l’argument
n_jobs=-1
.
import numpy as np
from sklearn.model_selection import GridSearchCV
import time
= time.time()
start_time # Parameters of pipelines can be set using ‘__’ separated parameter names:
= {
param_grid "boosting__n_estimators": np.linspace(5, 25, 5).astype(int),
"boosting__max_depth": [2, 4],
}= GridSearchCV(pipe, param_grid=param_grid)
grid_search
grid_search.fit(X_train, y_train)
= time.time()
end_time
print(f"Elapsed time : {int(end_time - start_time)} seconds")
GridSearchCV(estimator=Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])), ('boosting', GradientBoostingRegressor(random_state=0))]), param_grid={'boosting__max_depth': [2, 4], 'boosting__n_estimators': array([ 5, 10, 15, 20, 25])})In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
GridSearchCV(estimator=Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])), ('boosting', GradientBoostingRegressor(random_state=0))]), param_grid={'boosting__max_depth': [2, 4], 'boosting__n_estimators': array([ 5, 10, 15, 20, 25])})
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])), ('boosting', GradientBoostingRegressor(random_state=0))])
ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])
['Nombre_pieces_principales', 'surface']
SimpleImputer()
StandardScaler()
['month']
OneHotEncoder(handle_unknown='ignore', sparse_output=False)
GradientBoostingRegressor(random_state=0)
grid_search.best_params_ grid_search.best_estimator_
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])), ('boosting', GradientBoostingRegressor(max_depth=4, n_estimators=25, random_state=0))])In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
Pipeline(steps=[('preprocessor', ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])), ('boosting', GradientBoostingRegressor(max_depth=4, n_estimators=25, random_state=0))])
ColumnTransformer(transformers=[('pipeline', Pipeline(steps=[('simpleimputer', SimpleImputer()), ('standardscaler', StandardScaler())]), ['Nombre_pieces_principales', 'surface']), ('onehotencoder', OneHotEncoder(handle_unknown='ignore', sparse_output=False), ['month'])])
['Nombre_pieces_principales', 'surface']
SimpleImputer()
StandardScaler()
['month']
OneHotEncoder(handle_unknown='ignore', sparse_output=False)
GradientBoostingRegressor(max_depth=4, n_estimators=25, random_state=0)
Toutes les performances sur les ensembles d’échantillons et de test sur la grille d’hyperparamètres sont disponibles dans l’attribut:
= pd.DataFrame(grid_search.cv_results_) perf_random_forest
Regardons les résultats moyens pour chaque valeur des hyperparamètres:
= plt.subplots(1)
fig, ax = sns.lineplot(
g =perf_random_forest,
data=ax,
ax="param_boosting__n_estimators",
x="mean_test_score",
y="param_boosting__max_depth",
hue
)set(xlabel="Number of estimators", ylabel="Mean score on test sample")
g.
g
plt.legend(=(1.05, 1), loc="upper left", borderaxespad=0, title="Depth of trees"
bbox_to_anchor )
Globalement, à profondeur d’arbre donnée, le nombre d’arbres affecte la performance. Changer la profondeur de l’arbre améliore la performance de manière plus marquée.
Maintenant, il nous reste à re-entraîner le modèle avec ces nouveaux paramètres sur l’ensemble du jeu de train et l’évaluer sur l’ensemble du jeu de test :
= grid_search.best_estimator_
pipe_optimal
pipe_optimal.fit(X_train, y_train)
= pd.DataFrame([y_test, pipe_optimal.predict(X_test)]).T
compar = ["obs", "pred"]
compar.columns "diff"] = compar.obs - compar.pred compar[
On obtient le RMSE suivant :
Le RMSE sur le jeu de test est 562,317
Et si on regarde la qualité en prédiction:
On obtient plus de variance dans la prédiction, c’est déjà un peu mieux. Cependant, cela reste décevant pour plusieurs raisons:
- nous n’avons pas fait d’étape de sélection de variable
- nous n’avons pas chercher à déterminer si la variable à prédire la plus pertinente était le prix ou une transformation de celle-ci (par exemple le prix au \(m^2\))
6 Prochaine étape
Nous avons un modèle certes perfectible mais fonctionnel. La question qui se pose maintenant c’est d’essayer d’en faire quelque chose au service des utilisateurs. Cela nous amène vers la question de la mise en production.
Ceci est l’objet du prochain chapitre. Il s’agira d’une version introductive des enjeux évoqués dans le cadre du cours de 3e année de mise en production de projets de data science.
7 Références
Informations additionnelles
environment files have been tested on.
Latest built version: 2024-08-29
Python version used:
'3.11.6 | packaged by conda-forge | (main, Oct 3 2023, 10:40:35) [GCC 12.3.0]'
Package | Version |
---|---|
affine | 2.4.0 |
aiobotocore | 2.12.2 |
aiohttp | 3.9.3 |
aioitertools | 0.11.0 |
aiosignal | 1.3.1 |
alembic | 1.13.1 |
aniso8601 | 9.0.1 |
annotated-types | 0.7.0 |
appdirs | 1.4.4 |
archspec | 0.2.3 |
astroid | 3.1.0 |
asttokens | 2.4.1 |
attrs | 23.2.0 |
babel | 2.16.0 |
bcrypt | 4.1.2 |
beautifulsoup4 | 4.12.3 |
black | 24.8.0 |
blinker | 1.7.0 |
blis | 0.7.11 |
bokeh | 3.4.0 |
boltons | 23.1.1 |
boto3 | 1.34.51 |
botocore | 1.34.51 |
branca | 0.7.1 |
Brotli | 1.1.0 |
bs4 | 0.0.2 |
cachetools | 5.3.3 |
cartiflette | 0.0.2 |
Cartopy | 0.23.0 |
catalogue | 2.0.10 |
cattrs | 24.1.0 |
certifi | 2024.2.2 |
cffi | 1.16.0 |
charset-normalizer | 3.3.2 |
chromedriver-autoinstaller | 0.6.4 |
click | 8.1.7 |
click-plugins | 1.1.1 |
cligj | 0.7.2 |
cloudpathlib | 0.18.1 |
cloudpickle | 3.0.0 |
colorama | 0.4.6 |
comm | 0.2.2 |
commonmark | 0.9.1 |
conda | 24.3.0 |
conda-libmamba-solver | 24.1.0 |
conda-package-handling | 2.2.0 |
conda_package_streaming | 0.9.0 |
confection | 0.1.5 |
contextily | 1.6.1 |
contourpy | 1.2.1 |
cryptography | 42.0.5 |
cycler | 0.12.1 |
cymem | 2.0.8 |
cytoolz | 0.12.3 |
dask | 2024.4.1 |
dask-expr | 1.0.10 |
debugpy | 1.8.1 |
decorator | 5.1.1 |
dill | 0.3.8 |
distributed | 2024.4.1 |
distro | 1.9.0 |
docker | 7.0.0 |
duckdb | 0.10.1 |
en-core-web-sm | 3.7.1 |
entrypoints | 0.4 |
et-xmlfile | 1.1.0 |
exceptiongroup | 1.2.0 |
executing | 2.0.1 |
fastjsonschema | 2.19.1 |
fiona | 1.9.6 |
flake8 | 7.0.0 |
Flask | 3.0.2 |
folium | 0.16.0 |
fontawesomefree | 6.6.0 |
fonttools | 4.51.0 |
frozenlist | 1.4.1 |
fsspec | 2023.12.2 |
GDAL | 3.8.4 |
gensim | 4.3.2 |
geographiclib | 2.0 |
geopandas | 0.12.2 |
geoplot | 0.5.1 |
geopy | 2.4.1 |
gitdb | 4.0.11 |
GitPython | 3.1.43 |
google-auth | 2.29.0 |
graphene | 3.3 |
graphql-core | 3.2.3 |
graphql-relay | 3.2.0 |
graphviz | 0.20.3 |
great-tables | 0.10.0 |
greenlet | 3.0.3 |
gunicorn | 21.2.0 |
h11 | 0.14.0 |
htmltools | 0.5.3 |
hvac | 2.1.0 |
idna | 3.6 |
imageio | 2.35.1 |
importlib_metadata | 7.1.0 |
importlib_resources | 6.4.0 |
inflate64 | 1.0.0 |
ipykernel | 6.29.3 |
ipython | 8.22.2 |
ipywidgets | 8.1.2 |
isort | 5.13.2 |
itsdangerous | 2.1.2 |
jedi | 0.19.1 |
Jinja2 | 3.1.3 |
jmespath | 1.0.1 |
joblib | 1.3.2 |
jsonpatch | 1.33 |
jsonpointer | 2.4 |
jsonschema | 4.21.1 |
jsonschema-specifications | 2023.12.1 |
jupyter-cache | 1.0.0 |
jupyter_client | 8.6.1 |
jupyter_core | 5.7.2 |
jupyterlab_widgets | 3.0.10 |
kaleido | 0.2.1 |
kiwisolver | 1.4.5 |
kubernetes | 29.0.0 |
langcodes | 3.4.0 |
language_data | 1.2.0 |
lazy_loader | 0.4 |
libmambapy | 1.5.7 |
llvmlite | 0.42.0 |
locket | 1.0.0 |
lxml | 5.3.0 |
lz4 | 4.3.3 |
Mako | 1.3.2 |
mamba | 1.5.7 |
mapclassify | 2.6.1 |
marisa-trie | 1.2.0 |
Markdown | 3.6 |
markdown-it-py | 3.0.0 |
MarkupSafe | 2.1.5 |
matplotlib | 3.8.3 |
matplotlib-inline | 0.1.6 |
mccabe | 0.7.0 |
mdurl | 0.1.2 |
menuinst | 2.0.2 |
mercantile | 1.2.1 |
mizani | 0.11.4 |
mlflow | 2.11.3 |
mlflow-skinny | 2.11.3 |
msgpack | 1.0.7 |
multidict | 6.0.5 |
multivolumefile | 0.2.3 |
munkres | 1.1.4 |
murmurhash | 1.0.10 |
mypy | 1.9.0 |
mypy-extensions | 1.0.0 |
nbclient | 0.10.0 |
nbformat | 5.10.4 |
nest_asyncio | 1.6.0 |
networkx | 3.3 |
nltk | 3.8.1 |
numba | 0.59.1 |
numpy | 1.26.4 |
oauthlib | 3.2.2 |
opencv-python-headless | 4.9.0.80 |
openpyxl | 3.1.5 |
outcome | 1.3.0.post0 |
OWSLib | 0.28.1 |
packaging | 23.2 |
pandas | 2.2.1 |
paramiko | 3.4.0 |
parso | 0.8.4 |
partd | 1.4.1 |
pathspec | 0.12.1 |
patsy | 0.5.6 |
Pebble | 5.0.7 |
pexpect | 4.9.0 |
pickleshare | 0.7.5 |
pillow | 10.3.0 |
pip | 24.0 |
pkgutil_resolve_name | 1.3.10 |
platformdirs | 4.2.0 |
plotly | 5.19.0 |
plotnine | 0.13.6 |
pluggy | 1.4.0 |
polars | 0.20.31 |
preshed | 3.0.9 |
prometheus_client | 0.20.0 |
prometheus-flask-exporter | 0.23.0 |
prompt-toolkit | 3.0.42 |
protobuf | 4.25.3 |
psutil | 5.9.8 |
ptyprocess | 0.7.0 |
pure-eval | 0.2.2 |
py7zr | 0.20.8 |
pyarrow | 15.0.0 |
pyarrow-hotfix | 0.6 |
pyasn1 | 0.5.1 |
pyasn1-modules | 0.3.0 |
pybcj | 1.0.2 |
pycodestyle | 2.11.1 |
pycosat | 0.6.6 |
pycparser | 2.21 |
pycryptodomex | 3.20.0 |
pydantic | 2.8.2 |
pydantic_core | 2.20.1 |
pyflakes | 3.2.0 |
Pygments | 2.17.2 |
PyJWT | 2.8.0 |
pylint | 3.1.0 |
PyNaCl | 1.5.0 |
pynsee | 0.1.8 |
pyOpenSSL | 24.0.0 |
pyparsing | 3.1.2 |
pyppmd | 1.1.0 |
pyproj | 3.6.1 |
pyshp | 2.3.1 |
PySocks | 1.7.1 |
python-dateutil | 2.9.0 |
python-dotenv | 1.0.1 |
python-magic | 0.4.27 |
pytz | 2024.1 |
pyu2f | 0.1.5 |
pywaffle | 1.1.1 |
PyYAML | 6.0.1 |
pyzmq | 25.1.2 |
pyzstd | 0.16.1 |
QtPy | 2.4.1 |
querystring-parser | 1.2.4 |
rasterio | 1.3.10 |
referencing | 0.34.0 |
regex | 2023.12.25 |
requests | 2.31.0 |
requests-cache | 1.2.1 |
requests-oauthlib | 2.0.0 |
rich | 13.8.0 |
rpds-py | 0.18.0 |
rsa | 4.9 |
Rtree | 1.2.0 |
ruamel.yaml | 0.18.6 |
ruamel.yaml.clib | 0.2.8 |
s3fs | 2023.12.2 |
s3transfer | 0.10.1 |
scikit-image | 0.24.0 |
scikit-learn | 1.4.1.post1 |
scipy | 1.13.0 |
seaborn | 0.13.2 |
selenium | 4.24.0 |
setuptools | 69.2.0 |
shapely | 2.0.3 |
shellingham | 1.5.4 |
six | 1.16.0 |
smart_open | 7.0.4 |
smmap | 5.0.0 |
sniffio | 1.3.1 |
snuggs | 1.4.7 |
sortedcontainers | 2.4.0 |
soupsieve | 2.5 |
spacy | 3.7.6 |
spacy-legacy | 3.0.12 |
spacy-loggers | 1.0.5 |
SQLAlchemy | 2.0.29 |
sqlparse | 0.4.4 |
srsly | 2.4.8 |
stack-data | 0.6.2 |
statsmodels | 0.14.1 |
tabulate | 0.9.0 |
tblib | 3.0.0 |
tenacity | 8.2.3 |
texttable | 1.7.0 |
thinc | 8.2.5 |
threadpoolctl | 3.4.0 |
tifffile | 2024.8.28 |
tomli | 2.0.1 |
tomlkit | 0.12.4 |
toolz | 0.12.1 |
topojson | 1.9 |
tornado | 6.4 |
tqdm | 4.66.2 |
traitlets | 5.14.2 |
trio | 0.26.2 |
trio-websocket | 0.11.1 |
truststore | 0.8.0 |
typer | 0.12.5 |
typing_extensions | 4.11.0 |
tzdata | 2024.1 |
Unidecode | 1.3.8 |
url-normalize | 1.4.3 |
urllib3 | 1.26.18 |
wasabi | 1.1.3 |
wcwidth | 0.2.13 |
weasel | 0.4.1 |
webdriver-manager | 4.0.2 |
websocket-client | 1.8.0 |
Werkzeug | 3.0.2 |
wheel | 0.43.0 |
widgetsnbextension | 4.0.10 |
wordcloud | 1.9.3 |
wrapt | 1.16.0 |
wsproto | 1.2.0 |
xgboost | 2.0.3 |
xlrd | 2.0.1 |
xyzservices | 2024.4.0 |
yarl | 1.9.4 |
yellowbrick | 1.5 |
zict | 3.0.0 |
zipp | 3.17.0 |
zstandard | 0.22.0 |
View file history
SHA | Date | Author | Description |
---|---|---|---|
d242257 | 2024-08-22 18:51:51 | Lino Galiana | At this point, notebooks should now all be functional ! (#547) |
c641de0 | 2024-08-22 11:37:13 | Lino Galiana | A series of fix for notebooks that were bugging (#545) |
0908656 | 2024-08-20 16:30:39 | Lino Galiana | English sidebar (#542) |
8f0d2e1 | 2024-05-07 15:06:45 | lgaliana | Duplicate data from datagouv |
c9f9f8a | 2024-04-24 15:09:35 | Lino Galiana | Dark mode and CSS improvements (#494) |
06d003a | 2024-04-23 10:09:22 | Lino Galiana | Continue la restructuration des sous-parties (#492) |
8c316d0 | 2024-04-05 19:00:59 | Lino Galiana | Fix cartiflette deprecated snippets (#487) |
005d89b | 2023-12-20 17:23:04 | Lino Galiana | Finalise l’affichage des statistiques Git (#478) |
3fba612 | 2023-12-17 18:16:42 | Lino Galiana | Remove some badges from python (#476) |
4cd44f3 | 2023-12-11 17:37:50 | Antoine Palazzolo | Relecture NLP (#474) |
1684220 | 2023-12-02 12:06:40 | Antoine Palazzolo | Première partie de relecture de fin du cours (#467) |
e4642ee | 2023-11-27 17:02:05 | Lino Galiana | Deploy ML model as API (#460) |
a06a268 | 2023-11-23 18:23:28 | Antoine Palazzolo | 2ème relectures chapitres ML (#457) |
4960f2b | 2023-11-22 12:02:32 | Lino Galiana | Chapitre pipeline scikit sur DVF (#454) |
69cf52b | 2023-11-21 16:12:37 | Antoine Palazzolo | [On-going] Suggestions chapitres modélisation (#452) |
889a71b | 2023-11-10 11:40:51 | Antoine Palazzolo | Modification TP 3 (#443) |
652009d | 2023-10-09 13:56:34 | Lino Galiana | Finalise le cleaning (#430) |
a771183 | 2023-10-09 11:27:45 | Antoine Palazzolo | Relecture TD2 par Antoine (#418) |
154f09e | 2023-09-26 14:59:11 | Antoine Palazzolo | Des typos corrigées par Antoine (#411) |
3bdf3b0 | 2023-08-25 11:23:02 | Lino Galiana | Simplification de la structure 🤓 (#393) |
78ea2cb | 2023-07-20 20:27:31 | Lino Galiana | Change titles levels (#381) |
29ff3f5 | 2023-07-07 14:17:53 | linogaliana | description everywhere |
f21a24d | 2023-07-02 10:58:15 | Lino Galiana | Pipeline Quarto & Pages 🚀 (#365) |
38693f6 | 2023-04-19 17:22:36 | Lino Galiana | Rebuild visualisation part (#357) |
3248633 | 2023-02-18 13:11:52 | Lino Galiana | Shortcode rawhtml (#354) |
f10815b | 2022-08-25 16:00:03 | Lino Galiana | Notebooks should now look more beautiful (#260) |
494a85a | 2022-08-05 14:49:56 | Lino Galiana | Images featured ✨ (#252) |
d201e3c | 2022-08-03 15:50:34 | Lino Galiana | Pimp la homepage ✨ (#249) |
12965ba | 2022-05-25 15:53:27 | Lino Galiana | :launch: Bascule vers quarto (#226) |
9c71d6e | 2022-03-08 10:34:26 | Lino Galiana | Plus d’éléments sur S3 (#218) |
ed2ddec | 2022-01-04 13:32:43 | Expressso | retire typo (#212) |
66e2837 | 2021-12-24 16:54:45 | Lino Galiana | Fix a few typos in the new pipeline tutorial (#208) |
e94c1c5 | 2021-12-23 21:34:46 | Lino Galiana | Un tutoriel sur les pipelines :tada: (#203) |
Les références
Breiman, Leo. 1996. « Bagging predictors ». Machine learning 24: 123‑40.
———. 2001. « Random forests ». Machine learning 45: 5‑32.
Notes de bas de page
Les random forest sont l’une des principales méthodes ensemblistes. Outre cette approche, les plus connues sont le bagging (boostrap aggregating) et le boosting qui consistent à choisir la prédiction à privilégier selon des algorithmes de choix différens. Par exemple le bagging est une technique basée sur le vote majoritaire (Breiman 1996). Cette technique s’inspire du bootstrap qui, en économétrie, consiste à ré-estimer sur K sous-échantillons aléatoires des données un estimateur afin d’en tirer, par exemple, un intervalle de confiance empirique à 95%. Le principe du bagging est le même. On ré-estime K fois notre estimateur (par exemple un arbre de décision) et propose une règle d’agrégation pour en tirer une règle moyennisée et donc une prédiction. Le boosting fonctionne selon un principe différent, basé sur l’optimisation de combinaisons de classifieurs faibles.↩︎
Citation
BibTeX
@book{galiana2023,
author = {Galiana, Lino},
title = {Python pour la data science},
date = {2023},
url = {https://pythonds.linogaliana.fr/},
doi = {10.5281/zenodo.8229676},
langid = {fr}
}
Veuillez citer ce travail comme suit :
Galiana, Lino. 2023. Python pour la data science. https://doi.org/10.5281/zenodo.8229676.
2 Comment créer un pipeline
Un pipeline est un enchaînement d’opérations qu’on code en enchainant des pairs (clé, valeur) :
transform
et éventuellement une transformation inverse).On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
Au sein d’une étape de pipeline, les paramètres d’un estimateur sont accessibles avec la notation
<estimator>__<parameter>
. Cela permet de fixer des valeurs pour les arguments des fonctionsscikit
qui sont appelées au sein d’un pipeline. C’est cela qui rendra l’approche des pipelines particulièrement utile pour la grid search :On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
Ces pipelines sont initialisés sans données, il s’agit d’une structure formelle que nous allons ensuite ajuster en entraînant des modèles.
2.1 Données utilisées
Nous allons utiliser les données de transactions immobilières DVF pour chercher la meilleure manière de prédire, sachant les caractéristiques d’un bien, son prix.
Ces données sont mises à disposition sur
data.gouv
. Néanmoins, le format csv n’étant pas pratique pour importer des jeux de données volumineux, nous proposons de privilégier la versionParquet
mise à disposition par Eric Mauvière surdata.gouv
. L’approche la plus efficace pour lire ces données est d’utiliserDuckDB
afin de lire le fichier, extraire les colonnes d’intérêt puis passer àPandas
(pour en savoir plus sur l’intérêt deDuckDB
pour lire des fichiers volumineux, vous pouvez consulter ce post de blog ou celui-ci écrit par Eric Mauvière).Même si, en soi, les gains de temps sont faibles car
DuckDB
optimise les requêtes HTTPS nécessaires à l’import des données, nous proposons de télécharger les données pour réduire les besoins de bande passante.En premier lieu, puisque cela va faciliter les requêtes SQL ultérieures, on crée une vue :
Les données prennent la forme suivante :
Les variables que nous allons conserver sont les suivantes, nous allons les reformater pour la suite de l’exercice.
Note
Le fichier
Parquet
mis à disposition surdata.gouv
présente une incohérence de mise en forme de certaines colonnes à cause des virgules qui empêchent le formattage sous forme de colonne numérique.Le code ci-dessus effectue la conversion adéquate au niveau de
Pandas
.Introduire un effet confinement
Si vous travaillez avec les données de 2020, n’oubliez pas d’intégrer l’effet confinement dans vos modèles puisque cela a lourdement affecté les possibilités de transaction sur cette période, donc l’effet potentiel de certaines variables explicatives du prix.
Pour introduire cet effet, vous pouvez créer une variable indicatrice entre les dates en question:
Comme nous travaillons sur les données de 2022, nous pouvons nous passer de cette variable.
Les données DVF proposent une observation par transaction. Ces transactions peuvent concerner plusieurs lots. Par exemple, un appartement avec garage et cave comportera trois lots.
Pour simplifier, on va créer une variable de surface qui agrège les différentes informations de surface disponibles dans le jeu de données. Les agréger revient à supposer que le modèle de fixation des prix est le même entre chaque lot. C’est une hypothèse simplificatrice qu’une personne plus experte du marché immobilier, ou qu’une approche propre de sélection de variable pourrait amener à nier. En effet, les variables en question sont faiblement corrélées les unes entre elles, à quelques exceptions près (Figure 2.1):