7 juillet 2023

Syntaxe de base de la Classe

En langage orienté objet, une classe est un modèle de code programme extensible servant à créer des objets. Elle fournit les valeurs initiales de l’état (les variables membres) et de l’implémentation du comportement (les fonctions ou méthodes membres).

Wikipedia

En pratique, nous avons souvent besoin de créer beaucoup d’objets de même type, tels que des utilisateurs, des biens ou toute autre chose.

Comme nous le savons dans le chapitre Le constructeur, l'opérateur "new", new function peut nous aider à faire cela.

Mais dans le JavaScript moderne, il y a une construction de la “classe” plus avancée qui introduit de nombreux nouveaux aspects utiles en langage orienté objet.

La syntaxe de “classe”

La syntaxe de base est :

class MyClass {
  // Les méthodes de la classe
  constructor() { ... }
  method1() { ... }
  method2() { ... }
  method3() { ... }
  ...
}

Vous pouvez ensuite utiliser new MyClass() pour créer un nouvel objet ayant toute la liste des méthodes.

La méthode constructor() est automatiquement appelée par new, donc nous pouvons initialiser l’objet à ce niveau.

Par exemple :

class User {

  constructor(name) {
    this.name = name;
  }

  sayHi() {
    alert(this.name);
  }

}

// Usage:
let user = new User("John");
user.sayHi();

Lorsque new User("John") est appelé :

  1. Un nouvel objet est créé.
  2. Le constructor s’exécute avec les arguments qui lui sont passés et assigne this.name a l’objet.

…ensuite nous pouvons appeler les méthodes de l’objet, tel que user.sayHi().

Pas de virgule entre les méthodes de la classe

Un piège fréquent des développeurs novices est de mettre une virgule entre les méthodes de la classe, entrainant ainsi une erreur syntaxique.

La notation ici ne doit pas être confondue avec les objets littéraux. A l’intérieure d’une classe, aucune virgule n’est requise.

Qu’est-ce qu’une classe ?

Alors, c’est quoi exactement une class ? Ce n’est pas totalement une nouvelle entité au niveau du langage, comme on pourrait le penser.

Dévoilons maintenant la magie et regardons ce qu’est réellement une classe. Cela va nous aider à comprendre plusieurs aspects complexes.

En JavaScript, une classe est une sorte de fonction.

Regardons ici :

class User {
  constructor(name) { this.name = name; }
  sayHi() { alert(this.name); }
}

// La preuve : User est une fonction
alert(typeof User); // function

Ce que la construction de la classe classe User {...} fait réellement, c’est :

  1. Créer une fonction nommée User, qui devient le résultat de la déclaration de la classe. Le code de la fonction est tirée de la méthode constructor (considérée comme étant vide au cas ou cette méthode n’est pas écrite).
  2. Garde les méthodes de la classe, telle que sayHi, dans User.prototype.

Après la création de new User, lorsque nous appelons sa méthode, elle est extraite du prototype, comme décrit dans le chapitre F.prototype. Donc, l’objet a accès aux méthodes de classe.

Nous pouvons illustrer le résultat de la déclaration de class User ainsi :

Voici le code pour une introspection :

class User {
  constructor(name) { this.name = name; }
  sayHi() { alert(this.name); }
}

// classe est une fonction
alert(typeof User); // function

// ...ou, plus précisément, le constructeur de la méthode
alert(User === User.prototype.constructor); // true

// Les méthodes sont dans User.prototype, par exemple :
alert(User.prototype.sayHi); // le code de la méthode sayHi

// Il y a exactement deux méthodes dans le prototype
alert(Object.getOwnPropertyNames(User.prototype)); // constructeur, sayHi

Pas simplement un sucre syntaxique

Parfois certaines personnes disent que la notion de class est un “sucre syntaxique” (une syntaxe qui est destinée à rendre la lecture plus facile, mais elle n’introduit rien de nouveau), parce qu’en réalité nous pouvons déclarer la même chose sans utiliser le mot clé class :

// Réécriture de class User en fonctions pures

// 1. Créer la fonction constructeur
function User(name) {
  this.name = name;
}
// un prototype de fonction a une propriété constructeur par défaut,
// nous n'avons donc pas besoin de le créer

// 2. Ajouter la méthode au prototype
User.prototype.sayHi = function() {
  alert(this.name);
};

// Usage:
let user = new User("John");
user.sayHi();

Le résultat de cette définition est à peu près la même chose. Donc, il y a bien des raisons de vouloir considérer class comme pouvant être un sucre syntaxique pour définir un constructeur ensemble avec ses méthodes de prototype.

Cependant, il existe des différences importantes.

  1. Premièrement, une fonction créée par class est labellisée par une propriété interne spéciale [[IsClassConstructor]]: true. Ce n’est donc pas tout à fait la même chose que de le créer manuellement.

    Le langage vérifie cette propriété à divers endroits. Par exemple, contrairement à une fonction régulière, elle doit être appelée avec new :

    class User {
      constructor() {}
    }
    
    alert(typeof User); // fonction
    User(); // Erreur: le constructeur Class User ne peut être invoque sans 'new'

    Aussi, la représentation en chaîne de caractères d’un constructeur de class dans la plupart des moteurs de JavaScript commence avec “class…” :

    class User {
      constructor() {}
    }
    
    alert(User); // class User { ... }

    Il y a d’autres différences, nous les verrons bientôt.

  2. Les méthodes de Class sont non-énumérable. Une définition de la classe attribue false à la propriété enumerable pour toutes les méthodes du "prototype".

    C’est bien, parce que si nous exécutons un for..in sur un Object, souvent nous ne voulons pas accéder aux méthodes de sa classe.

  3. Les Classes utilisent toujours use strict. Tout code à l’intérieur de la construction de la classe est automatiquement en mode strict.

En outres, la syntaxe classe apporte beaucoup d’autres caractéristiques que nous allons explorer plus tard.

L’Expression Class

Tout comme les fonctions, les classes peuvent être définies a l’intérieur d’une autre expression, passées en paramètres, retournées, assignées etc.

Voici un exemple d’expression d’une classe :

let User = class {
  sayHi() {
    alert("Hello");
  }
};

Similairement aux Fonction Expressions nommées, les expressions de classe peuvent avoir un nom.

Si une expression de classe a un nom, il est visible à l’intérieur de la classe uniquement :

// "Expression de Classe nommée"
// (Terme non existant dans la spécification, mais elle est similaire a une Expression de Fonction nommée)
let User = class MyClass {
  sayHi() {
    alert(MyClass); // le nom MyClass est seulement visible dans la classe
  }
};

new User().sayHi(); // ça fonctionne, montre la définition de MyClass

alert(MyClass); // erreur, le nom MyClass n'est pas visible en dehors de la classe

Nous pouvons même créer les classes dynamiquement “à la demande”, comme suit :

function makeClass(phrase) {
  // déclare une classe et la retourne
  return class {
    sayHi() {
      alert(phrase);
    }
  };
}

// Crée une nouvelle classe
let User = makeClass("Hello");

new User().sayHi(); // Hello

Accesseurs/Mutateurs

Tout comme les objets littéraux, les classes peuvent inclure des accesseurs/mutateurs, des propriétés évaluées etc.

Voici un exemple pour user.name implémenté en utilisant les propriétés get/set :

class User {

  constructor(name) {
    // invoque l'accesseur (the setter)
    this.name = name;
  }

  get name() {
    return this._name;
  }

  set name(value) {
    if (value.length < 4) {
      alert("Name is too short.");
      return;
    }
    this._name = value;
  }

}

let user = new User("John");
alert(user.name); // John

user = new User(""); // le nom est trop court.

Techniquement, une telle déclaration de classe fonctionne en créant des getters et des setters dans User.prototype.

Computed names […]

Voici un exemple avec un nom de méthode calculé utilisant des crochets [...] :

class User {

  ['say' + 'Hi']() {
    alert("Hello");
  }

}

new User().sayHi();

Ces caractéristiques sont faciles à retenir, car elles ressemblent à celles d’objets littéraux.

Champs de classe

Les anciens navigateurs peuvent avoir besoin de polyfill

Les propriétés de classe sont un ajout récent au langage.

Auparavant, nos classes n’avaient que des méthodes.

Les “Class fields” (champs de classe) sont une syntaxe qui permet d’ajouter des propriétés.

Par exemple, ajoutons la propriété name à class User :

class User {
  name = "John";

  sayHi() {
    alert(`Hello, ${this.name}!`);
  }
}

new User().sayHi(); // Hello, John!

Il suffit donc d’écrire “ = ” dans la déclaration, et c’est tout.

La différence importante des champs de classe est qu’ils sont définis sur des objets individuels, et non sur User.prototype :

class User {
  name = "John";
}

let user = new User();
alert(user.name); // John
alert(User.prototype.name); // undefined

Nous pouvons également attribuer des valeurs à l’aide d’expressions et d’appels de fonctions plus complexes :

class User {
  name = prompt("Name, please?", "John");
}

let user = new User();
alert(user.name); // John

Création de méthodes liées avec des champs de classe

Comme démontré dans le chapitre Le "bind" de fonction les fonctions en JavaScript ont un this dynamique. Cela dépend du contexte de l’appel.

Donc, si une méthode objet est contournée et appelée dans un autre contexte, this ne sera plus une référence à son objet.

Par exemple, ce code affichera undefined :

class Button {
  constructor(value) {
    this.value = value;
  }

  click() {
    alert(this.value);
  }
}

let button = new Button("hello");

setTimeout(button.click, 1000); // undefined

Le problème est appelé “perdre le this”.

Il existe deux approches pour le corriger, comme indiqué dans le chapitre Le "bind" de fonction :

  1. Passer une fonction wrapper, telle que setTimeout(() => button.click(), 1000).
  2. Lier la méthode à l’objet, par exemple dans le constructeur.

Les champs de classe fournissent une syntaxe plus élégante :

class Button {
  constructor(value) {
    this.value = value;
  }
  click = () => {
    alert(this.value);
  }
}

let button = new Button("hello");

setTimeout(button.click, 1000); // hello

Le champ de classe click = () => {...} est créé par objet, il y a une fonction distincte pour chaque objet Button, avec this à l’intérieur référençant cet objet. Nous pouvons passer button.click n’importe où, et la valeur de this sera toujours correcte.

C’est particulièrement utile dans un environnement de navigateur, pour les écouteurs d’événements.

Résumé

La syntaxe de base d’une classe ressemble à ceci :

class MyClass {
  prop = value; // propriété

  constructor(...) { // constructeur
    // ...
  }

  method(...) {} // méthode

  get something(...) {} //  méthode définie avec un accesseur
  set something(...) {} //  méthode définie avec un mutateur

  [Symbol.iterator]() {} // méthode avec un nom évalué (symbole ici)
  // ...
}

MyClass est techniquement une fonction (celle que nous fournissons en tant que constructor), tandis que les méthodes, accesseurs et mutateurs sont écrits dans MyClass.prototype.

Dans les prochains chapitres nous apprendrons plus à propos des classes, y compris la notion d’héritage et les autres caractéristiques.

Exercices

importance: 5

La classe Clock (voir la sandbox) est écrite en style fonctionnelle. Réécrivez la en syntaxe de “classe”.

P.S. La montre doit tictaquer dans la console, ouvrez la pour la voir.

Open a sandbox for the task.

class Clock {
  constructor({ template }) {
    this.template = template;
  }

  render() {
    let date = new Date();

    let hours = date.getHours();
    if (hours < 10) hours = '0' + hours;

    let mins = date.getMinutes();
    if (mins < 10) mins = '0' + mins;

    let secs = date.getSeconds();
    if (secs < 10) secs = '0' + secs;

    let output = this.template
      .replace('h', hours)
      .replace('m', mins)
      .replace('s', secs);

    console.log(output);
  }

  stop() {
    clearInterval(this.timer);
  }

  start() {
    this.render();
    this.timer = setInterval(() => this.render(), 1000);
  }
}


let clock = new Clock({template: 'h:m:s'});
clock.start();

Ouvrez la solution dans une sandbox.

Carte du tutoriel