TP plus proches voisins : Classification des survivants du Titanic#

On souhaite prédire si un passager du Titanic a survécu ou non à l’accident, en utilisant l’algorithme des plus proches voisins. On pourra s’inspirer de l’exemple du cours sur la classification de fleurs.

Voici les informations sur chaque passager :

  • Survived : 0 = Non, 1 = Oui

  • Pclass : Classe de ticket (1 = 1ère classe, 2 = 2ème, 3 = 3ème)

  • Sex : Genre du passager (male ou female)

  • Age : Âge du passager (en années)

  • Fare : Tarif du ticket (en dollars)

Chargement des données avec Pandas#

Pandas est un module Python qui permet de manipuler des données sous forme de tableau appelé DataFrame (qui ressemble à un peu à une table SQL).

Pour charger les données dans Basthon :

  1. Télécharger les données (clic droit ici puis enregistrer sous).

  2. Dans Basthon, cliquer sur Fichier puis Ouvrir et sélectionner le fichier téléchargé.

  3. Exécuter le code ci-dessous, en modifiant titanic.csv si vous avez utilisé un autre nom de fichier.

import pandas as pd

df = pd.read_csv('titanic.csv') # df est un DataFrame
df.head() # pour afficher les 5 premières lignes
Survived Pclass Sex Age Fare
0 0 3 male 22.0 7.2500
1 1 1 female 38.0 71.2833
2 1 3 female 26.0 7.9250
3 1 1 female 35.0 53.1000
4 0 3 male 35.0 8.0500

Ainsi df est un tableau contient 5 colonnes (Survived, Pclass, Sex, Age, Fare) et chaque ligne correspondant à un passager du Titanic. On peut obtenir le nombres de lignes avec len(df) :

len(df)
891

Chaque ligne est identifiée par un index (= nom de la ligne), ici \(0, 1, 2, ...\). On peut accéder à la ligne d’indice \(i\) avec df.loc[i] :

df.loc[0]
Survived       0
Pclass         3
Sex         male
Age         22.0
Fare        7.25
Name: 0, dtype: object

df.loc[i] donne en fait une Series, qui peut être vu comme un tableau (à une dimension).

On peut récupérer une colonne (également sous forme de series), par exemple Age, avec df["Age"] :

df["Age"]
0      22.0
1      38.0
2      26.0
3      35.0
4      35.0
       ... 
886    27.0
887    19.0
888    28.0
889    26.0
890    32.0
Name: Age, Length: 891, dtype: float64

On peut combiner ces deux méthodes pour récupérer une valeur précise, par exemple l’âge du 3ème passager :

df.loc[2, "Age"]
26.0

On peut aussi modifier une valeur avec, par exemple, df.loc[2, "Age"] = ....

On peut parcourir les indices d’un dataframe avec df.index. Par exemple, pour trouver le passager le plus vieux :

maxi_age = 0
for i in df.index:
    if df.loc[i, "Age"] > maxi_age:
        maxi_age = df.loc[i, "Age"]
maxi_age
80.0

Remarque : avec Pandas, il faut normalement utiliser au maximum des opérations vectorielles pour que le processeur puisse effectuer les calculs en parallèle. Cependant, comme l’utilisation de Pandas n’est pas au programme, nous allons nous limiter à une approche élémentaire.

Statistiques#

Question

Écrire une fonction moyenne(df, c) qui renvoie la moyenne des valeurs sur la colonne c du dataframe df. Quelle est l’âge moyen des passagers du Titanic ? Le prix moyen du ticket ?

print("Âge moyen :", moyenne(df, "Age"))
print("Prix moyen du billet :", moyenne(df, "Fare"))
Âge moyen : 29.36158249158249
Prix moyen du billet : 32.2042079685746

Question

Écrire une fonction ecart_type(df, c) qui renvoie l’écart-type des valeurs de la colonne c du dataframe df. On rappelle que l’écart-type d’une série de valeurs \(x_1, \ldots, x_n\) est donné par :

\[\sqrt{\frac{1}{n} \sum_{i=1}^n (x_i - \bar{x})^2}\]

\(\bar{x}\) est la moyenne des valeurs \(x_1, ..., x_n\).

Remarque : On évitera de calculer plusieurs fois la même moyenne.

ecart_type(df, "Age")
13.01238827279366

Question

Afficher le pourcentage de survivants parmi :

  • les hommes

  • les femmes

  • les passagers de 1ère classe

  • les passagers de 3ème classe

for c, v in [("Sex", "male"), ("Sex", "female"), ("Pclass", 1), ("Pclass", 3)]:
    print("Taux de survie pour", c, "=", v, ":", survivants(c, v))
Taux de survie pour Sex = male : 0.18890814558058924
Taux de survie pour Sex = female : 0.7420382165605095
Taux de survie pour Pclass = 1 : 0.6296296296296297
Taux de survie pour Pclass = 3 : 0.24236252545824846

Variables catégorielles#

Nous souhaitons modéliser chaque passager par un vecteur de \(\mathbb{R}^4\) (car il y a \(4\) informations pour chaque passager : âge, genre, classe et prix du ticket). Cependant, le genre est une variable catégorielle qu’il faut transformer en variable numérique :

df["Sex"] = df["Sex"].map({"male": 0, "female": 1}) # remplace male par 0 et female par 1
df.head()
Survived Pclass Sex Age Fare
0 0 3 0 22.0 7.2500
1 1 1 1 38.0 71.2833
2 1 3 1 26.0 7.9250
3 1 1 1 35.0 53.1000
4 0 3 0 35.0 8.0500

Standardisation#

On remarque que les attributs sont sur des échelles très différentes (par exemple, l’âge est entre 0 et 80, alors que la classe du billet est entre 1 et 3).
Les différences d’âge contribuent alors beaucoup plus dans les calculs de distance, ce qui ferait que l’âge aurait un poids plus important que la classe du billet pour la prédiction.
Pour éviter cela, on va standardiser les données, c’est-à-dire les transformer de manière à ce que chaque attribut ait une moyenne nulle et un écart-type égal à 1.

Si un attribut \(x\) a une moyenne \(\bar{x}\) et un écart-type \(\sigma\), on peut le standardiser en le remplaçant par :

\[\frac{x - \bar{x}}{\sigma}\]

Question

Écrire une fonction standardiser(df, c) qui standardise la colonne c du dataframe df. L’utiliser pour standardiser les colonnes Age, Fare, Pclass et Sex. On rappelle qu’on peut modifier l’élément sur la ligne i et la colonne c avec df.loc[i, c] = ....

for c in ["Age", "Fare", "Pclass", "Sex"]:
    standardiser(df, c)
df.head()
Survived Pclass Sex Age Fare
0 0 0.827377 -0.737695 -0.565736 -0.502445
1 1 -1.566107 1.355574 0.663861 0.786845
2 1 0.827377 1.355574 -0.258337 -0.488854
3 1 -1.566107 1.355574 0.433312 0.420730
4 0 0.827377 -0.737695 0.433312 -0.486337

Distance#

Pour la question suivante, on rappelle comment accéder aux attributs d’une donnée :

p = df.loc[0] # 1er passager
p["Age"], p["Fare"], p["Pclass"], p["Sex"] # attributs de p
(-0.565736461074875,
 -0.5024451714361915,
 0.8273772438659676,
 -0.7376951317802897)

Question

Écrire une fonction distance(p1, p2) qui calcule la distance euclidienne entre les passagers p1 et p2. On prendra en compte tous les attributs sauf Survived.

distance(df.loc[0], df.loc[1]) # distance entre les deux premiers passagers
3.644820996221396

Séparation des données#

On sépare les données en deux : une partie train utilisée pour la prédiction, et une partie test utilisée pour évaluer la qualité de la prédiction :

train = df.sample(frac=0.9,random_state=0)
test = df.drop(train.index)
print("nombre de données dans train :", len(train))
print("nombre de données dans test :", len(test))
nombre de données dans train : 802
nombre de données dans test : 89

Algorithmes des plus proches voisins#

Question

Écrire une fonction voisins(x, k) qui renvoie les indices des \(k\) plus proches voisins de x dans train.

voisins(test.iloc[0], 5)
[446, 651, 546, 427, 389]

Question

Écrire une fonction plus_frequent(L) qui renvoie l’élément le plus fréquent d’une liste L.

plus_frequent([2, 1, 5, 1, 2, 5, 5])
5

Question

Écrire une fonction knn(x, k) qui renvoie la prédiction de survie de x en utilisant l’algorithme des \(k\) plus proches voisins.

knn(test.iloc[0], 5)
1

Analyse des résultats#

Question

Écrire une fonction precision(k) qui renvoie la précision de l’algorithme des \(k\) plus proches voisins en utilisant k voisins.
Remarque : cela peut prendre quelques secondes.

precision(3)
0.8314606741573034

Question

Écrire une fonction plot_precision(kmax) qui trace la précision pour \(k\) variant de \(1\) à kmax. Quelle est la meilleure précision obtenue pour k entre 1 et 5 (cela prend environ 1 minute) ? Quelle est le nombre de voisins optimal ?

Kaggle#

Ce TP provient d’une compétition Kaggle : Titanic: Machine Learning from Disaster.
Avec KNN, j’obtiens un score de \(\approx 0.763\)… Ce qui n’est pas terrible car l’exemple de submission basée uniquement sur le genre donne un score de \(\approx 0.765\).

Question

Soumettre le fichier submission.csv obtenu par le code ci-dessus sur Kaggle et essayer d’obtenir un score plus élevé. On pourra essayer d’implémenter les améliorations suggérées à la fin du cours.

submission = pd.read_csv('test.csv')
for c in "Age", "Fare":
    submission[c].fillna(submission[c].median(), inplace=True)
submission["Sex"] = submission["Sex"].map({"male": 0, "female": 1})
for c in ["Age", "Fare", "Pclass", "Sex"]:
    standardiser(submission, c)
    
submission["Survived"] = [knn(submission.iloc[i], 5) for i in range(len(submission))]

submission[["PassengerId", "Survived"]].to_csv('submission.csv', index=False)