ES6 of the Week

This Week's Episode:

maps and sets

Maps and Sets

  • In JavaScript, objects wear many hats:
    • We use objects as HashMaps or dictionaries, as well as instances of classes
  • Sometimes, if we know we want to store data in a certain way, it may be safer and more efficient to use a different data structure than the JS engine's implementation of objects
  • Enter: Maps and Sets

Sets

  • Sets are a collection of unique values of any type
    • This includes object references
  • Items in a Set will be iterated through in the order they were inserted
  • Because each item in the set is unique, you can check their equality at a high level (i.e. if you compare an item in a Set that references a specific object in memory, it will evaluate as equal to another pointer to that same object in memory

On your mark, get set...

let _set = new Set(); // n.b. set is a reserved word, so I'm calling my variable _set

// Set.prototype.add
_set.add("string").add(42);

// store object references
let obj = { a : 1 };
_set.add(obj);

// Set.prototype.size
_set.size // => 3
_set.add(42) // no repeats
_set.size // => 3

// Set.prototype.has
_set.has("string") // true
_set.has(obj) // true, wowzers!
_set.has(21) // false

// Iterate using for...of
for (let item of _set) console.log(item); // logs in insertion order

// Set.prototype.delete and Set.prototype.clear
_set.delete("string"); // remove a single item
_set.clear(); // wipe the set clean

Arrays

Could have duplicates

 

Actually an object

 

Iterate with for loop

Sets

Unique values

 

No baggage

 

Iterate with for...of

Maps

  • A key-value map
  • Keys can contain object references, functions, or whatever
  • No need to worry about Object.hasOwnProperty when iterating
  • Has handy methods that make it easy to get the size, or iterate over either the values or keys

Objects

Has a prototype (default key)

 

Keys are strings or symbols

 

Cannot get size easily

Maps

No prototype

 

Keys are any type

 

Can get size easily

Follow the Map!

let _map = new Map(); // now I'm just being consistent ;)

// Map.prototype.set
_map.set('a', 1);
_map.set('b', 2);

let objKey = { 'c' : 3 };
_map.set(objKey, 4);

// Map.prototype.get
_map.get('a'); // => 1
_map.get(objKey); // => 4

// Map.prototype.size - much easier than using Object.hasOwnProperty!
_map.size; // => 3

// Iterate over keys using Map.prototype.keys
for (let key of _map.keys()) console.log(key);

// Iterate over values using Map.prototype.values
for (let value of _map.values()) console.log(value);

WeakSet and WeakMap

  • Have a subset of the same Set/Map methods/properties (but not all)
    • WS: Add, has, delete
    • WM: Get, set, has delete
  • WeakSets can only contain objects, and the keys of WeakMaps can only be objects
  • Objects in WeakSets and WeakMaps are weakly held, meaning that if no other reference to them exists, they will be garbage collected
    • Advantageous for freeing up memory
    • This also means they are not enumerable (not iterable)

WeakMap example

let _counter = new WeakMap();
let _action = new WeakMap();

class Countdown {
    constructor(counter, action) {
        _counter.set(this, counter);
        _action.set(this, action);
    }
    dec() {
        let counter = _counter.get(this);
        if (counter < 1) return;
        counter--;
        _counter.set(this, counter);
        if (counter === 0) {
            _action.get(this)();
        }
    }
}

// http://www.2ality.com/2015/01/es6-maps-sets.html

Use Cases

  • Set
    • Any collection of unique values
  • Map
    • Quite versatile
    • Dictionary-like collections, especially those that will be iterated over
  • WeakSet
    • 'branding' your classes?
  • WeakMap
    • storing metadata for an object
    • prevent memory leaks in the DOM
    • http://stackoverflow.com/questions/29413222/what-are-the-actual-uses-of-es6-weakmap

Resources

  • ES6: http://es6-features.org/
  • Wiki: https://en.wikipedia.org/wiki/Object_(computer_science)
  • Mozilla docs:
    • https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set
    • https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map
    • https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakSet
    • https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap
  • Blogs & StackOverflow
    • http://bjorn.tipling.com/maps-sets-and-iterators-in-javascript
    • http://www.sitepoint.com/preparing-ecmascript-6-set-weakset/
    • http://stackoverflow.com/questions/29413222/what-are-the-actual-uses-of-es6-weakmap
    • https://esdiscuss.org/topic/actual-weakset-use-cases
    • http://www.2ality.com/2015/01/es6-maps-sets.html

ES6 of the Week - 5

By Tom Kelly

ES6 of the Week - 5

Maps and Sets

  • 1,587