30. Maps (Map)
Before ES6, JavaScript didn’t have a data structure for dictionaries and (ab)used objects as dictionaries from strings to arbitrary values. ES6 brought Maps, which are dictionaries from arbitrary values to arbitrary values.
30.1. Using Maps
An instance of Map
maps keys to values. A single key-value mapping is called an entry. Maps record in which order entries were created and honor that order when returning, e.g., keys or entries.
30.1.1. Creating Maps
There are three common ways of creating Maps.
First, you can use the constructor without any parameters to create an empty Map:
Second, you can pass an iterable (e.g. an Array) over key-value “pairs” (Arrays with 2 elements) to the constructor:
Third, the .set()
method adds entries to a Map and is chainable:
30.1.2. Working with single entries
.set()
and .get()
are for writing and reading values (given keys).
.has()
checks if a Map has an entry with a given key. .delete()
removes entries.
30.1.3. Determining the size of a Map and clearing it
.size
contains the number of entries in a Map. .clear()
removes all entries of a Map.
30.1.4. Getting the keys and values of a Map
.keys()
returns an iterable over the keys of a Map:
We can use spreading (…
) to convert the iterable returned by .keys()
to an Array:
.values()
works like .keys()
, but for values instead of keys.
30.1.5. Getting the entries of a Map
.entries()
returns an iterable over the entries of a Map:
Spreading (…
) converts the iterable returned by .entries()
to an Array:
Map instances are also iterables over entries. In the following code, we use destructuring to access the keys and values of map
:
30.2. Example: Counting characters
countChars()
returns a Map that maps characters to numbers of occurrences.
function countChars(chars) {
const charCounts = new Map();
for (let ch of chars) {
ch = ch.toLowerCase();
const prevCount = charCounts.get(ch) || 0;
charCounts.set(ch, prevCount+1);
}
return charCounts;
}
const result = countChars('AaBccc');
assert.deepEqual(
[...result],
[
['a', 2],
['b', 1],
['c', 3],
]
);
30.3. A few more details about the keys of Maps (advanced)
Any value can be a key, even an object:
30.3.1. What keys are considered equal?
Most Map operations need to check whether a value is equal to one of the keys. They do so via the internal operation SameValueZero, which works like ===
, but considers NaN
to be equal to itself.
As a consequence, you can use NaN
as a key in Maps, just like any other value:
Different objects are always considered to be different. That is something that can’t be configured (yet – TC39 is aware that this is important functionality).
30.4. Missing Map operations
30.4.1. Mapping and filtering Maps
You can .map()
and .filter()
Arrays, but there are no such operations for Maps. The solution is:
- Convert the Map into an Array of [key,value] pairs.
- Map or filter the Array.
- Convert the result back to a Map.
I’ll use the following Map to demonstrate how that works.
Mapping originalMap
:
Filtering originalMap
:
Step 1 is performed by the spread operator (…
).
30.4.2. Combining Maps
There are no methods for combining Maps, which is why the approach from the previous section must be used to do so.
Let’s combine the following two Maps:
To combine map1
and map2
, we turn them into Arrays via the spread operator (…
) and concatenate those Arrays. Afterwards, we convert the result back to a Map. All of that is done in line A.
30.5. Quick reference: Map<K,V>
Note: For the sake of conciseness, I’m pretending that all keys have the same type K
and that all values have the same type V
.
30.5.1. Constructor
new Map<K, V>(entries?: Iterable<[K, V]>)
[ES6]
If you don’t provide the parameter entries
then an empty Map is created. If you do provide an iterable over [key, value] pairs then those pairs are used to add entries to the Map. For example:
30.5.2. Map<K,V>.prototype: handling single entries
.get(key: K): V
[ES6]
Returns the value
that key
is mapped to in this Map. If there is no key key
in this Map, undefined
is returned.
.set(key: K, value: V): this
[ES6]
Maps the given key to the given value. If there is already an entry whose key is key
, it is updated. Otherwise, a new entry is created. This method returns this
, which means that you can chain it.
.has(key: K): boolean
[ES6]
Returns whether the given key exists in this Map.
.delete(key: K): boolean
[ES6]
If there is an entry whose key is key
, it is removed and true
is returned. Otherwise, nothing happens and false
is returned.
30.5.3. Map<K,V>.prototype: handling all entries
get .size: number
[ES6]
Returns how many entries there are in this Map.
In JavaScript indexable sequences (such as Arrays) have a .length
, while mainly unordered collections (such as Maps) have a .size
.
.clear(): void
[ES6]
Removes all entries from this Map.
30.5.4. Map<K,V>.prototype: iterating and looping
Both iterating and looping happen in the order in which entries were added to a Map.
.entries(): Iterable<[K,V]>
[ES6]
Returns an iterable with one [key,value] pair for each entry in this Map. The pairs are Arrays of length 2.
.forEach(callback: (value: V, key: K, theMap: Map<K,V>) => void, thisArg?: any): void
[ES6]
The first parameter is a callback that is invoked once for each entry in this Map. If thisArg
is provided, this
is set to it for each invocation. Otherwise, this
is set to undefined
.
.keys(): Iterable<K>
[ES6]
Returns an iterable over all keys in this Map.
.values(): Iterable<V>
[ES6]
Returns an iterable over all values in this Map.
Symbol.iterator: Iterable<[K,V]>
[ES6]
The default way of iterating over Maps. Same as .entries()
.
30.5.5. Sources
30.6. FAQ
30.6.1. When should I use a Map, when an object?
If you map anything other than strings to any kind of data, you have no choice: you must use a Map.
If, however, you are mapping strings to arbitrary data, you must decide whether or not to use an object. A rough general guideline is:
Is there a fixed set of keys (known at development time)?Then use an object and access the values via fixed keys:
obj.key
Can the set of keys change at runtime?Then use a Map and access the values via keys stored in variables:
map.get(theKey)
30.6.2. When would I use an object as a key in a Map?
Map keys mainly make sense if they are compared by value (the same “content” means that two values are considered equal, not the same identity). That excludes objects. There is one use case – externally attaching data to objects, but that use case is better served by WeakMaps where entries don’t prevent garbage collection (for details, consult the next chapter).