Jusquâà présent, nous avons découvert les structures de données complexes suivantes :
- Les objets sont utilisés pour stocker des collections de clés.
- Les tableaux sont utilisés pour stocker des collections ordonnées.
Mais ce nâest pas suffisant pour la vie réelle. Câest pourquoi Map et Set existent également.
Map
Une Map est une collection dâéléments de données saisis, tout comme un Object. Mais la principale différence est que Map autorise les clés de tout type.
Voici les méthodes et les propriétés dâune Map :
new Map()â créer la map.map.set(key, value)â stocke la valeur par la clé.map.get(key)â renvoie la valeur par la clé,undefinedsikeynâexiste pas dans la map.map.has(key)â retournetruesi lakeyexiste,falsesinon.map.delete(key)â supprime lâélément (la paire clé/valeur) par la clé.map.clear()â supprime tout de la map.map.sizeâ renvoie le nombre dâéléments actuel.
Par exemple :
let map = new Map();
map.set('1', 'str1'); // une clé de type chaîne de caractère
map.set(1, 'num1'); // une clé de type numérique
map.set(true, 'bool1'); // une clé de type booléenne
// souvenez-vous, dans un `Object`, les clés sont converties en chaîne de caractères
// alors que `Map` conserve le type d'origine de la clé,
// c'est pourquoi les deux appels suivants retournent des valeurs :
alert( map.get(1) ); // 'num1'
alert( map.get('1') ); // 'str1'
alert( map.size ); // 3
Au travers de cet exemple nous pouvons voir, quâà la différence des Objects, les clés ne sont pas converties en chaîne de caractère.
Il est donc possible dâutiliser nâimporte quel type.
map[key] nâest pas la bonne façon dâutiliser un MapBien que map[key] fonctionne également, par exemple nous pouvons définir map[key] = 2, cela traite map comme un objet JavaScript simple, ce qui implique toutes les limitations correspondantes (uniquement des clés chaîne de caractères/symbol etc.).
Nous devons donc utiliser les méthodes de map : set, get et ainsi de suite.
Map peut également utiliser des objets comme clés.
Par exemple :
let john = { name: "John" };
// pour chaque utilisateur, nous stockons le nombre de visites
let visitsCountMap = new Map();
// john est utilisé comme clé dans la map
visitsCountMap.set(john, 123);
alert( visitsCountMap.get(john) ); // 123
Utiliser des objets comme clés est lâune des fonctionnalités les plus notables et les plus importantes de Map. La même chose ne compte pas pour Object. Une chaîne de caractères comme clé dans Object est très bien, mais nous ne pouvons pas utiliser un autre Object comme clé dans Object.
Essayons de faire comme lâexemple précédent directement avec un Object :
let john = { name: "John" };
let ben = { name: "Ben" };
let visitsCountObj = {}; // on créé notre object
visitsCountObj[ben] = 234; // essayez d'utiliser l'objet ben comme clé
visitsCountObj[john] = 123; // essayez d'utiliser l'objet john comme clé, l'objet ben sera remplacé
// C'est ce qui a été écrit!
alert( visitsCountObj["[object Object]"] ); // 123
Comme visitesCountObj est un objet, il convertit toutes les clés Object, telles que john et ben ci-dessus, en la même chaîne de caractères "[object Object]". Certainement pas ce que nous voulons.
Map compare les clésPour tester lâégalité entre les clés, Map se base sur lâalgorithme SameValueZero.
Câest grosso modo la même chose que lâopérateur de stricte égalité ===, à la différence que NaN est considéré comme étant égal à NaN.
NaN peut donc être utilisé comme clé.
Cet algorithme ne peut pas être modifié.
Chaque appel à map.set retourne la map elle-même, ce qui nous permet dâenchaîner les appels :
map.set('1', 'str1')
.set(1, 'num1')
.set(true, 'bool1');
Itération dans Map
Il existe 3 façons de parcourir les éléments dâune map :
map.keys()â renvoie un itérable pour les clés,map.values()â renvoie un itérable pour les valeurs,map.entries()â renvoie un itérable pour les entrées[key, value], il est utilisé par défaut dansfor..of.
Par exemple :
let recipeMap = new Map([
['cucumber', 500],
['tomatoes', 350],
['onion', 50]
]);
// on parcourt les clés (les légumes)
for (let vegetable of recipeMap.keys()) {
alert(vegetable); // cucumber, tomatoes, onion
}
// on parcourt les valeurs (les montants)
for (let amount of recipeMap.values()) {
alert(amount); // 500, 350, 50
}
// on parcourt les entries (couple [clé, valeur])
for (let entry of recipeMap) { // équivalent à : recipeMap.entries()
alert(entry); // cucumber,500 (etc.)
}
Contraitement aux Object, Map conserve lâordre dâinsertion des valeurs.
Il est aussi possible dâutiliser forEach avec Map comme on pourrait le faire avec un tableau :
// exécute la fonction pour chaque couple (key, value)
recipeMap.forEach( (value, key, map) => {
alert(`${key}: ${value}`); // cucumber: 500 etc.
});
Object.entries: Créer une Map à partir dâun objet
Lorsquâune Map est créée, nous pouvons passer un tableau (ou un autre itérable) contenant des paires clé/valeur pour lâinitialisation, comme ceci :
// tableau de paires [clé, valeur]
let map = new Map([
['1', 'str1'],
[1, 'num1'],
[true, 'bool1']
]);
alert( map.get('1') ); // str1
Si nous avons un objet simple et que nous souhaitons en créer une Map, nous pouvons utiliser la méthode intégrée Object.entries(obj) qui renvoie un tableau de paires clé/valeur pour un objet exactement dans ce format.
Nous pouvons donc créer une Map à partir dâun objet de la manière suivante :
let obj = {
name: "John",
age: 30
};
let map = new Map(Object.entries(obj));
alert( map.get('name') ); // John
Ici, Object.entries renvoie le tableau de paires clé/valeur : [ ["name", "John"], ["age", 30] ]. Câest ce dont a besoin la Map.
Object.fromEntries: Objet à partir dâune Map
Nous venons de voir comment créer une Map à partir dâun objet simple avec Object.entries(obj).
Il existe une méthode Object.fromEntries qui fait lâinverse : étant donné un tableau de paires [clé, valeur], elle crée un objet à partir de ces paires :
let prices = Object.fromEntries([
['banana', 1],
['orange', 2],
['meat', 4]
]);
// maintenant, prices = { banana: 1, orange: 2, meat: 4 }
alert(prices.orange); // 2
Nous pouvons utiliser Object.fromEntries pour obtenir un objet simple à partir dâune Map.
Par exemple, nous stockons les données dans une Map, mais nous devons les transmettre à un code tiers qui attend un objet simple.
Voici comment procéder :
let map = new Map();
map.set('banana', 1);
map.set('orange', 2);
map.set('meat', 4);
let obj = Object.fromEntries(map.entries()); // créer un objet simple (*)
// terminé!
// obj = { banana: 1, orange: 2, meat: 4 }
alert(obj.orange); // 2
Un appel à map.entries() renvoie un itérable de paires clé/valeur, exactement dans le bon format pour Object.fromEntries.
Nous pourrions également raccourcir la ligne (*) :
let obj = Object.fromEntries(map); // .entries() omis
Câest la même chose, car Object.fromEntries attend un objet itérable en argument. Pas nécessairement un tableau. Et lâitération standard pour une map renvoie les mêmes paires clé/valeur que map.entries(). Ainsi, nous obtenons un objet simple avec les mêmes clés/valeurs que la map.
Set
Un Set est une collection de types spéciaux â âensemble de valeursâ (sans clés), où chaque valeur ne peut apparaître quâune seule fois.
Ses principales méthodes sont :
new Set([iterable])â crée le set et si un objetiterableest fourni (généralement un tableau), en copie les valeurs dans le set.set.add(value)â ajoute une valeur, renvoie le set lui-même.set.delete(value)â supprime la valeur, renvoietruesivalueexistait au moment de lâappel, sinonfalse.set.has(value)â renvoietruesi la valeur existe dans le set sinonfalse.set.clear()â supprime tout du set.set.sizeâ câest le nombre dâéléments.
Ce quâil faut surtout savoir câest que lorsque lâon appelle plusieurs fois set.add(value) avec la même valeur, la méthode ne fait rien.
Câest pourquoi chaque valeur est unique dans un Set.
Par exemple, nous souhaitons nous souvenir de tous nos visiteurs. Mais chaque visiteurs doit être unique.
Set est exactement ce quâil nous faut :
let set = new Set();
let john = { name: "John" };
let pete = { name: "Pete" };
let mary = { name: "Mary" };
// visites, certains utilisateurs viennent plusieurs fois
set.add(john);
set.add(pete);
set.add(mary);
set.add(john);
set.add(mary);
// set conserve une fois chaque visiteurs
alert( set.size ); // 3
for (let user of set) {
alert(user.name); // John (puis Pete et Mary)
}
Lâalternative à Set aurait pu être un tableau dâutilisateurs en vérifiant avant chaque insertion que lâélément nâexiste pas en utilisant arr.find. Cependant les performances auraient été moins bonnes car cette méthode parcours chaque élément du tableau. Set est beaucoup plus efficace car il est optimisé en interne pour vérifier lâunicité des valeurs.
Parcourir un Set
Nous pouvons parcourir les éléments dâun Set avec for..of ou en utilisant forEach :
let set = new Set(["oranges", "apples", "bananas"]);
for (let value of set) alert(value);
// même chose en utilisant forEach:
set.forEach((value, valueAgain, set) => {
alert(value);
});
A noter que la fonction de callback utilisée par forEach prend 3 arguments en paramètres : une value, puis la même valeur valueAgain, et enfin le set lui-même.
Câest pour la compatibilité avec Map où le callback forEach passé possède trois arguments. Ãa a lâair un peu étrange, câest sûr. Mais cela peut aider à remplacer facilement Map par Set dans certains cas, et vice versa.
Les méthodes pour parcourir les éléments dâune Map peuvent être utilisées :
set.keys()â renvoie un objet itérable pour les valeurs,set.values()â identique Ãset.keys(), pour compatibilité avecMap,set.entries()â renvoie un objet itérable pour les entrées[value, value], existe pour la compatibilité avecMap.
Résumé
Map â est une collection de valeurs-clés.
Méthodes et propriétés :
new Map([iterable])â crée la map, avec uniterablefacultatif (par exemple un tableau) de paires[key, value]pour lâinitialisation.map.set(key, value)â stocke la valeur par la clé, renvoie la map elle-même.map.get(key)â renvoie la valeur par la clé,undefinedsikeynâexiste pas dans la map.map.has(key)â retournetruesi lakeyexiste,falsesinon.map.delete(key)â supprime lâélément par la clé, renvoietruesikeyexistait au moment de lâappel, sinonfalse.map.clear()â supprime tout de la map.map.sizeâ renvoie le nombre dâéléments actuel.
La différence entre Map avec un objet traditionel :
- Nâimporte quel type peut être utilisé comme clé.
- Accès à des méthodes tels que
size.
Set â est une collection de valeurs uniques.
Méthodes et propriétés :
new Set([iterable])â crée le set avec uniterablefacultatif (par exemple un tableau) de valeurs pour lâinitialisation.set.add(value)â ajoute une valeur (ne fait rien sivalueexiste), renvoie lâensemble lui-même.set.delete(value)â supprime la valeur, renvoietruesivalueexistait au moment de lâappel, sinonfalse.set.has(value)â renvoietruesi la valeur existe dans lâensemble, sinonfalse.set.clear()â supprime tout du set.set.sizeâ câest le nombre dâéléments.
On ne peut pas dire que les éléments dans une Map ou un Set sont désordonnés car ils sont toujours parcourut par ordre dâinsertion.
Il est cependant impossible de réorganiser les éléments ou bien de les retrouver par leur index.
Commentaires
<code>, pour plusieurs lignes â enveloppez-les avec la balise<pre>, pour plus de 10 lignes - utilisez une sandbox (plnkr, jsbin, codepenâ¦)