La syntaxe de décomposition permet d'étendre un itérable (par exemple une expression de tableau ou une chaîne de caractères) en lieu et place de plusieurs arguments (pour les appels de fonctions) ou de plusieurs éléments (pour les littéraux de tableaux) ou de paires clés-valeurs (pour les littéraux d'objets).

Syntaxe

Pour l'utilisation de la décomposition dans les appels de fonction :

f(...objetIterable);

Pour les littéraux de tableaux :

[...objetIterable, 4, 5, 6]

Pour la décomposition :

[a, b, ...objetIterable] = [1, 2, 3, 4, 5];

Pour les littéraux objets (nouvelle fonctionnalité pour ECMAScript, actuellement en proposition de niveau 4, finalisée) :

let objClone = { ...obj };

Exemples

Utiliser la décomposition dans les appels de fonction

Améliorer la fonction apply

Il arrive souvent qu'on veuille utiliser Function.prototype.apply avec un tableau parmi les arguments de la fonction utilisée :

function f(x, y, z) { }
var args = [0, 1, 2];
f.apply(null, args);

avec la décomposition, on peut désormais écrire :

function f(x, y, z) { }
var args = [0, 1, 2];
f(...args);

Tout argument passé à une fonction peut être décomposé grâce à cette syntaxe et cette syntaxe peut être utilisée pour plusieurs arguments.

function f(v, w, x, y, z) { }
var args = [0, 1];
f(-1, ...args, 2, ...[3]);

Utiliser apply avec l'opérateur new

Avec ES5, il n'est pas possible d'utiliser new avec apply (selon ES5 apply effectue un appel [[Call]] et pas un appel [[Construct]]). Avec ES2015, la syntaxe de décomposition permet de le faire naturellement :

var champsDate = lireChampsDate(maBaseDeDonnées);
var d = new Date(...champsDate);

Afin d'utiliser new avec un tableau de paramètres, sans utiliser la décomposition, il faudrait l'employer indirectement grâce à une application partielle :

function applyAndNew(constructor, args) {
   function partial () {
      return constructor.apply(this, args);
   };
   if (typeof constructor.prototype === "object") {
      partial.prototype = Object.create(constructor.prototype);
   }
   return partial;
}


function monConstructeur () {
   console.log("arguments.length: " + arguments.length);
   console.log(arguments);
   this.prop1="val1";
   this.prop2="val2";
};

var mesArguments = ["bi", "bop", "bup", null];
var monConstructeurAvecArguments = applyAndNew(monConstructor, mesArguments);

console.log(new monConstructeurAvecArguments);
// (log fourni par monConstructeur):           arguments.length: 4
// (log fourni par monConstructeur):           ["bi", "bop", "bup", null]
// (log fourni par "new monConstructeurAvecArguments"): {prop1: "val1", prop2: "val2"}

Améliorer la fonction push

push est souvent utilisé pour ajouter un tableau à la fin d'un autre tableau. Avec ES5, on peut utiliser le code suivant :

var arr1 = [0, 1, 2];
var arr2 = [3, 4, 5];
// On ajoute tous les éléments de arr2 à la fin de arr1
Array.prototype.push.apply(arr1, arr2);

Avec ES2015 et la décomposition, on peut utiliser :

var arr1 = [0, 1, 2];
var arr2 = [3, 4, 5];
arr1.push(...arr2);

Utiliser la décomposition dans les littéraux de tableau

Améliorer les littéraux de tableau

À l'heure actuelle, sans la décomposition, si on a un tableau et qu'on souhaite créer un nouveau tableau composé du premier, on ne peut pas utiliser un littéral de tableau et il faut utiliser des fonctions comme push, splice, concat, etc. Avec la syntaxe de décomposition, cela devient plus succinct :

var articulations = ['épaules', 'genoux'];
var corps = ['têtes', ...articulations, 'bras', 'pieds'];
// ["têtes", "épaules", "genoux", "bras", "pieds"]

Comme pour les fonctions, la syntaxe peut être utilisé plusieurs fois.

Copier un tableau

var arr = [1, 2, 3];
var arr2 = [...arr];
arr2.push(4);

console.log(arr2); // [1, 2, 3, 4]
console.log(arr);  // [1, 2, 3] (inchangé)

Note : Lorsqu'on utilise la décomposition pour copier un tableau, celle-ci ne s'applique qu'au premier niveau de profondeur. Par conséquent, il peut ne pas convenir pour la copie des tableaux multidimensionnels (des tableaux imbriqués dans d'autres tableaux) comme le montre l’exemple suivant (il en va de même avec Object.assign() et la décomposition).

var a = [[1], [2], [3]];
var b = [...a]; // b vaut [[1], [2], [3]]

b.shift().shift(); // *a* vaut désormais [[], [2], [3]];

Une meilleure façon de concaténer des tableaux

concat est souvent utilisé afin de concaténer un tableau à la suite d'une autre. Avec ES5, on aurait le code suivant :

var arr1 = [0, 1, 2];
var arr2 = [3, 4, 5];
// On ajoute les éléments de arr2 après ceux de arr1
var nouveauTableau = arr1.concat(arr2);

Avec ES2015 et la décomposition, on peut écrire :

var arr1 = [0, 1, 2];
var arr2 = [3, 4, 5];
arr1 = [...arr1, ...arr2];

unshift est souvent utilisé afin d'insérer des valeurs d'un tableau au début d'un autre tableau. Avec ES5, on peut écrire :

var arr1 = [0, 1, 2];
var arr2 = [3, 4, 5];
// On ajoute tous les éléments
// de arr2 au début de arr1
Array.prototype.unshift.apply(arr1, arr2) // arr1 vaut [3, 4, 5, 0, 1, 2]

Avec ES2015 et la décomposition, on peut écrire :

var arr1 = [4, 5, 6];
var arr2 = [1, 2, 3];
arr1 = [...arr2, ...arr1];
// arr1 vaut désormais [1, 2, 3, 4, 5, 6]

Note : Il y a une différence avec unshift() : ici, on crée un nouveau tableau qui est affecté à arr1, le tableau original de arr1 n'est pas modifié "sur place".

Utiliser la décomposition avec les littéraux objet

La proposition relative à la décomposition des propriétés (actuellement au stade de proposition de niveau 4) vise à ajouter la décomposition des propriétés pour les littéraux objets. Cela permet de copier les propriétés énumérables directement rattachées à un objet source sur un nouvel objet.

Le clonage superficiel (qui ne rattache pas le prototype) ou la fusion d'objets peut donc être obtenue avec une syntaxe plus concise que celle utilisant Object.assign().

var obj1 = { toto: 'truc', x: 42 };
var obj2 = { toto: 'bidule', y: 13 };

var clone = { ...obj1 };
// Object { toto: 'truc', x: 42 }

var fusion = { ...obj1, ...obj2 };
// Object { toto: 'bidule', x: 42, y: 13 };

On notera que Object.assign() déclenche les mutateurs, ce qui n'est pas le cas pour la syntaxe de décomposition.

Il n'est pas possible de remplacer ou de recopier le comportement de la fonction Object.assign() :

var obj1 = { toto: 'truc', x: 42 };
var obj2 = { toto: 'bidule', y: 51 };

const fusion = ( ...objets) => ({...objets});
var objFusionné = fusion(obj1, obj2);
// Object { 0: { toto: 'truc', x: 42 }, 1: { toto: 'bidule', y: 51 } }

var objFusionné2 = fusion({}, obj1, obj2);
// Object { 0: {}, 1: { toto: 'truc', x: 42 }, 2: { toto: 'bidule', y: 51 } }

Dans l'exemple précédent, la syntaxe de décomposition ne fonctionne pas comme on pourrait s'y attendre : il décompose les arguments en un tableau grâce au paramètre du reste.

La décomposition ne s'applique qu'aux itérables

Pour rappel : la syntaxe de décomposition ne s'applique qu'aux objets itérables :

var obj = {"clé1" : "valeur1"};
function maFonction(x) {
  console.log(x); // undefined
}
maFonction(...obj);
var args = [...obj];
console.log(args, args.length) //[] 0

Utiliser la décomposition avec de nombreuses valeurs

Lorsqu'on utilise la décomposition (comme dans les exemples précédents), il faut faire attention à ne pas dépasser le nombre maximal d'arguments du moteur JavaScript. En effet, la décomposition place toutes les valeurs sources dans la pile. Pour plus d'informations, consulter Function.prototype.apply.

Spécifications

Spécification État Commentaires
ECMAScript 2015 (6th Edition, ECMA-262) Standard Définie dans plusieurs sections de la spécification : initialisateur de tableau, listes d'argument.
ECMAScript 2018 (ECMA-262) Standard Définie dans la section sur les initialisateurs d'objet.
ECMAScript Latest Draft (ECMA-262) Projet Aucune modification.
ECMAScript Latest Draft (ECMA-262) Projet Aucune modification.

Compatibilité des navigateurs

Update compatibility data on GitHub
OrdinateurMobileServeur
ChromeEdgeFirefoxInternet ExplorerOperaSafariWebview AndroidChrome pour AndroidEdge MobileFirefox pour AndroidOpera pour AndroidSafari pour iOSSamsung InternetNode.js
Spread in array literalsChrome Support complet 46Edge Support complet 12Firefox Support complet 16IE Aucun support NonOpera Support complet 37Safari Support complet 8WebView Android Support complet 46Chrome Android Support complet 46Edge Mobile Support complet 12Firefox Android Support complet 16Opera Android Support complet 37Safari iOS Support complet 8Samsung Internet Android Support complet 5.0nodejs Support complet 5.0.0
Support complet 5.0.0
Support complet 4.0.0
Désactivée
Désactivée From version 4.0.0: this feature is behind the --harmony runtime flag.
Spread in function callsChrome Support complet 46Edge Support complet 12Firefox Support complet 27IE Aucun support NonOpera Support complet 37Safari Support complet 8WebView Android Support complet 46Chrome Android Support complet 46Edge Mobile Support complet 12Firefox Android Support complet 27Opera Android Support complet 37Safari iOS Support complet 8Samsung Internet Android Support complet 5.0nodejs Support complet 5.0.0
Support complet 5.0.0
Support complet 4.0.0
Désactivée
Désactivée From version 4.0.0: this feature is behind the --harmony runtime flag.
Spread in destructuringChrome Support complet 49Edge Aucun support NonFirefox Support complet 34IE Aucun support NonOpera Support complet 37Safari ? WebView Android Support complet 49Chrome Android Support complet 49Edge Mobile Aucun support NonFirefox Android Support complet 34Opera Android Support complet 37Safari iOS ? Samsung Internet Android Support complet 5.0nodejs Support complet Oui
Spread in object literals
Expérimentale
Chrome Support complet 60Edge Aucun support NonFirefox Support complet 55IE Aucun support NonOpera ? Safari Aucun support NonWebView Android Support complet 60Chrome Android Support complet 60Edge Mobile Aucun support NonFirefox Android Support complet 55Opera Android ? Safari iOS Aucun support NonSamsung Internet Android Aucun support Nonnodejs Support complet 8.3.0
Support complet 8.3.0
Support complet 8.0.0
Désactivée
Désactivée From version 8.0.0: this feature is behind the --harmony runtime flag.

Légende

Support complet  
Support complet
Aucun support  
Aucun support
Compatibilité inconnue  
Compatibilité inconnue
Fonctionnalité expérimentale. Celle-ci peut être amenée à changer par la suite.
Fonctionnalité expérimentale. Celle-ci peut être amenée à changer par la suite.
Une action explicite de l'utilisateur est nécessaire pour activer cette fonctionnalité.
Une action explicite de l'utilisateur est nécessaire pour activer cette fonctionnalité.

Voir aussi

Étiquettes et contributeurs liés au document

Contributeurs à cette page : SphinxKnight, edspeedy
Dernière mise à jour par : SphinxKnight,